mirror of
https://github.com/sbrl/Pepperminty-Wiki.git
synced 2025-01-09 19:34:56 +00:00
4817 lines
690 KiB
PHP
4817 lines
690 KiB
PHP
|
<?php
|
||
|
/**
|
||
|
* @license
|
||
|
*
|
||
|
* PHP Error
|
||
|
*
|
||
|
* Copyright (c) 2012 Joseph Lenton
|
||
|
* All rights reserved.
|
||
|
*
|
||
|
* Redistribution and use in source and binary forms, with or without
|
||
|
* modification, are permitted provided that the following conditions are met:
|
||
|
* * Redistributions of source code must retain the above copyright
|
||
|
* notice, this list of conditions and the following disclaimer.
|
||
|
* * Redistributions in binary form must reproduce the above copyright
|
||
|
* notice, this list of conditions and the following disclaimer in the
|
||
|
* documentation and/or other materials provided with the distribution.
|
||
|
* * Neither the name of the <organization> nor the
|
||
|
* names of its contributors may be used to endorse or promote products
|
||
|
* derived from this software without specific prior written permission.
|
||
|
*
|
||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
||
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||
|
* DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
|
||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||
|
*
|
||
|
* Uses:
|
||
|
* JSMin-php https://github.com/rgrove/jsmin-php/
|
||
|
* jQuery http://jquery.com/
|
||
|
*/
|
||
|
|
||
|
/**
|
||
|
* PHP Error
|
||
|
*
|
||
|
* --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- ---
|
||
|
*
|
||
|
* WARNING! It is downright _DANGEROUS_ to use this in production, on
|
||
|
* a live website. It should *ONLY* be used for development.
|
||
|
*
|
||
|
* PHP Error will kill your environment at will, clear the output
|
||
|
* buffers, and allows HTML injection from exceptions.
|
||
|
*
|
||
|
* In future versions it plans to do far more then that.
|
||
|
*
|
||
|
* If you use it in development, awesome! If you use it in production,
|
||
|
* you're an idiot.
|
||
|
*
|
||
|
* --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- ---
|
||
|
*
|
||
|
* = Info
|
||
|
*
|
||
|
* A small API for replacing the standard PHP errors, with prettier
|
||
|
* error reporting. This will change the error reporting level, and this
|
||
|
* is deliberate, as I believe in strict development errors.
|
||
|
*
|
||
|
* simple usage:
|
||
|
*
|
||
|
* \php_error\reportErrors();
|
||
|
*
|
||
|
* Advanced example:
|
||
|
*
|
||
|
* There is more too it if you want more customized error handling. You
|
||
|
* can pass in options, to customize the setup, and you get back a
|
||
|
* handler you can alter at runtime.
|
||
|
*
|
||
|
* $handler = new \php_error\ErrorHandler( $myOptions );
|
||
|
* $handler->turnOn();
|
||
|
*
|
||
|
* There should only ever be one handler! This is an (underdstandable)
|
||
|
* limitation in PHP. It's because if an exception or error is raised,
|
||
|
* then there is a single point of handling it.
|
||
|
*
|
||
|
* = INI Options
|
||
|
*
|
||
|
* - php_error.force_disabled When set to a true value (such as on),
|
||
|
* this forces this to be off.
|
||
|
* This is so you can disable this script
|
||
|
* in your production servers ini file,
|
||
|
* incase you accidentally upload this there.
|
||
|
*
|
||
|
* --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- ---
|
||
|
*
|
||
|
* @author Joseph Lenton | https://github.com/josephlenton
|
||
|
*/
|
||
|
|
||
|
namespace php_error;
|
||
|
|
||
|
use \php_error\FileLinesSet,
|
||
|
\php_error\ErrorHandler,
|
||
|
|
||
|
\php_error\JSMin,
|
||
|
\php_error\JSMinException;
|
||
|
|
||
|
use \Closure,
|
||
|
\Exception,
|
||
|
\ErrorException,
|
||
|
\InvalidArgumentException;
|
||
|
|
||
|
use \ReflectionMethod,
|
||
|
\ReflectionFunction,
|
||
|
\ReflectionParameter;
|
||
|
|
||
|
global $_php_error_already_setup,
|
||
|
$_php_error_global_handler,
|
||
|
$_php_error_is_ini_enabled;
|
||
|
|
||
|
/*
|
||
|
* Avoid being run twice.
|
||
|
*/
|
||
|
if ( empty($_php_error_already_setup) ) {
|
||
|
$_php_error_already_setup = true;
|
||
|
|
||
|
/*
|
||
|
* These are used as token identifiers by PHP.
|
||
|
*
|
||
|
* If they are missing, then they should never pop out of PHP,
|
||
|
* so we just give them their future value.
|
||
|
*
|
||
|
* They are primarily here so I don't have to alter the 5.3
|
||
|
* compliant code. Instead I can delete pre-5.3 code (this
|
||
|
* code), in the future.
|
||
|
*
|
||
|
* As long as the value is unique, and does not clash with PHP,
|
||
|
* then any number could be used. That is why they start counting
|
||
|
* at 100,000.
|
||
|
*/
|
||
|
|
||
|
$missingIdentifier = array(
|
||
|
'T_INSTEADOF',
|
||
|
'T_TRAIT',
|
||
|
'T_TRAIT_C',
|
||
|
'T_YIELD',
|
||
|
'T_FINALLY'
|
||
|
);
|
||
|
|
||
|
$counter = 100001;
|
||
|
foreach ( $missingIdentifier as $id ) {
|
||
|
if ( ! defined($id) ) {
|
||
|
define( $id, $counter++ );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Check if it's empty, in case this file is loaded multiple times.
|
||
|
*/
|
||
|
if ( ! isset($_php_error_global_handler) ) {
|
||
|
$_php_error_global_handler = null;
|
||
|
|
||
|
$_php_error_is_ini_enabled = false;
|
||
|
|
||
|
/*
|
||
|
* check both 'disable' and 'disabled' incase it's mispelt
|
||
|
* check that display errors is on
|
||
|
* and ensure we are *not* a command line script.
|
||
|
*/
|
||
|
$_php_error_is_ini_enabled =
|
||
|
! @get_cfg_var( 'php_error.force_disabled' ) &&
|
||
|
! @get_cfg_var( 'php_error.force_disable' ) &&
|
||
|
@ini_get('display_errors') === '1' &&
|
||
|
PHP_SAPI !== 'cli'
|
||
|
;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* This is shorthand for turning off error handling,
|
||
|
* calling a block of code, and then turning it on.
|
||
|
*
|
||
|
* However if 'reportErrors' has not been called,
|
||
|
* then this will silently do nothing.
|
||
|
*
|
||
|
* @param callback A PHP function to call.
|
||
|
* @return The result of calling the callback.
|
||
|
*/
|
||
|
function withoutErrors( $callback ) {
|
||
|
global $_php_error_global_handler;
|
||
|
|
||
|
if ( $_php_error_global_handler !== null ) {
|
||
|
return $_php_error_global_handler->withoutErrors( $callback );
|
||
|
} else {
|
||
|
return $callback();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Turns on error reporting, and returns the handler.
|
||
|
*
|
||
|
* If you just want error reporting on, then don't bother
|
||
|
* catching the handler. If you're building something
|
||
|
* clever, like a framework, then you might want to grab
|
||
|
* and use it.
|
||
|
*
|
||
|
* Note that calling this a second time will replace the
|
||
|
* global error handling with a new error handler.
|
||
|
* The existing one will be turned off, and the new one
|
||
|
* turned on.
|
||
|
*
|
||
|
* You can't use two at once!
|
||
|
*
|
||
|
* @param options Optional, options declaring how PHP Error should be setup and used.
|
||
|
* @return The ErrorHandler used for reporting errors.
|
||
|
*/
|
||
|
function reportErrors( $options=null ) {
|
||
|
$handler = new ErrorHandler( $options );
|
||
|
return $handler->turnOn();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* The actual handler. There can only ever be one.
|
||
|
*/
|
||
|
class ErrorHandler
|
||
|
{
|
||
|
const REGEX_DOCTYPE = '/<( )*!( *)DOCTYPE([^>]+)>/';
|
||
|
|
||
|
const REGEX_PHP_IDENTIFIER = '\b[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*';
|
||
|
const REGEX_PHP_CONST_IDENTIFIER = '/\b[A-Z_\x7f-\xff][A-Z0-9_\x7f-\xff]*/';
|
||
|
|
||
|
/**
|
||
|
* Matches:
|
||
|
* {closure}()
|
||
|
* blah::foo()
|
||
|
* foo()
|
||
|
*
|
||
|
* It is:
|
||
|
* a closure
|
||
|
* or a method or function
|
||
|
* followed by parenthesis '()'
|
||
|
*
|
||
|
* a function is 'namespace function'
|
||
|
* a method is 'namespace class::function', or 'namespace class->function'
|
||
|
* the whole namespace is optional
|
||
|
* namespace is made up of an '\' and then repeating 'namespace\'
|
||
|
* both the first slash, and the repeating 'namespace\', are optional
|
||
|
*
|
||
|
* 'END' matches it at the end of a string, the other one does not.
|
||
|
*/
|
||
|
const REGEX_METHOD_OR_FUNCTION_END = '/(\\{closure\\})|(((\\\\)?(\b[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*\\\\)*)?\b[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*(::[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*)?)\\(\\)$/';
|
||
|
const REGEX_METHOD_OR_FUNCTION = '/(\\{closure\\})|(((\\\\)?(\b[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*\\\\)*)?\b[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*(::[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*)?)\\(\\)/';
|
||
|
|
||
|
const REGEX_VARIABLE = '/\b[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*$/';
|
||
|
|
||
|
const REGEX_MISSING_SEMI_COLON_FOLLOWING_LINE = '/^ *(return|}|if|while|foreach|for|switch)/';
|
||
|
|
||
|
/**
|
||
|
* The number of lines to take from the file,
|
||
|
* where the error is reported. This is the number
|
||
|
* of lines around the line in question,
|
||
|
* including that line.
|
||
|
*
|
||
|
* So '9' will be the error line + 4 lines above + 4 lines below.
|
||
|
*/
|
||
|
const NUM_FILE_LINES = 13;
|
||
|
|
||
|
const FILE_TYPE_APPLICATION = 1;
|
||
|
const FILE_TYPE_IGNORE = 2;
|
||
|
const FILE_TYPE_ROOT = 3;
|
||
|
|
||
|
/*
|
||
|
* These are the various magic identifiers,
|
||
|
* used for headers, post requests, and so on.
|
||
|
*
|
||
|
* Their main purpose is to be long and more or less unique,
|
||
|
* enough that a collision with user code is rare.
|
||
|
*/
|
||
|
|
||
|
const PHP_ERROR_MAGIC_HEADER_KEY = 'PHP_ERROR_MAGIC_HEADER';
|
||
|
const PHP_ERROR_MAGIC_HEADER_VALUE = 'php_stack_error';
|
||
|
const MAGIC_IS_PRETTY_ERRORS_MARKER = '<!-- __magic_php_error_is_a_stack_trace_constant__ -->';
|
||
|
|
||
|
const HEADER_SAVE_FILE = 'PHP_ERROR_SAVE_FILES';
|
||
|
|
||
|
const POST_FILE_LOCATION = 'php_error_upload_file';
|
||
|
|
||
|
const PHP_ERROR_INI_PREFIX = 'php_error';
|
||
|
|
||
|
/**
|
||
|
* At the time of writing, scalar type hints are unsupported.
|
||
|
* By scalar, I mean 'string' and 'integer'.
|
||
|
*
|
||
|
* If they do get added, this is here as a trap to turn scalar
|
||
|
* type hint warnings on and off.
|
||
|
*/
|
||
|
private static $IS_SCALAR_TYPE_HINTING_SUPPORTED = false;
|
||
|
|
||
|
private static $SCALAR_TYPES = array(
|
||
|
'string', 'integer', 'float', 'boolean',
|
||
|
'bool', 'int', 'number'
|
||
|
);
|
||
|
|
||
|
/**
|
||
|
* A mapping of PHP internal symbols,
|
||
|
* mapped to descriptions of them.
|
||
|
*/
|
||
|
private static $PHP_SYMBOL_MAPPINGS = array(
|
||
|
'$end' => 'end of file',
|
||
|
'T_ABSTRACT' => 'abstract',
|
||
|
'T_AND_EQUAL' => "'&='",
|
||
|
'T_ARRAY' => 'array',
|
||
|
'T_ARRAY_CAST' => 'array cast',
|
||
|
'T_AS' => "'as'",
|
||
|
'T_BOOLEAN_AND' => "'&&'",
|
||
|
'T_BOOLEAN_OR' => "'||'",
|
||
|
'T_BOOL_CAST' => 'boolean cast',
|
||
|
'T_BREAK' => 'break',
|
||
|
'T_CASE' => 'case',
|
||
|
'T_CATCH' => 'catch',
|
||
|
'T_CLASS' => 'class',
|
||
|
'T_CLASS_C' => '__CLASS__',
|
||
|
'T_CLONE' => 'clone',
|
||
|
'T_CLOSE_TAG' => 'closing PHP tag',
|
||
|
'T_CONCAT_EQUAL' => "'.='",
|
||
|
'T_CONST' => 'const',
|
||
|
'T_CONSTANT_ENCAPSED_STRING' => 'string',
|
||
|
'T_CONTINUE' => 'continue',
|
||
|
'T_CURLY_OPEN' => '\'{$\'',
|
||
|
'T_DEC' => '-- (decrement)',
|
||
|
'T_DECLARE' => 'declare',
|
||
|
'T_DEFAULT' => 'default',
|
||
|
'T_DIR' => '__DIR__',
|
||
|
'T_DIV_EQUAL' => "'/='",
|
||
|
'T_DNUMBER' => 'number',
|
||
|
'T_DOLLAR_OPEN_CURLY_BRACES' => '\'${\'',
|
||
|
'T_DO' => "'do'",
|
||
|
'T_DOUBLE_ARROW' => "'=>'",
|
||
|
'T_DOUBLE_CAST' => 'double cast',
|
||
|
'T_DOUBLE_COLON' => "'::'",
|
||
|
'T_ECHO' => 'echo',
|
||
|
'T_ELSE' => 'else',
|
||
|
'T_ELSEIF' => 'elseif',
|
||
|
'T_EMPTY' => 'empty',
|
||
|
'T_ENCAPSED_AND_WHITESPACE' => 'non-terminated string',
|
||
|
'T_ENDDECLARE' => 'enddeclare',
|
||
|
'T_ENDFOR' => 'endfor',
|
||
|
'T_ENDFOREACH' => 'endforeach',
|
||
|
'T_ENDIF' => 'endif',
|
||
|
'T_ENDSWITCH' => 'endswitch',
|
||
|
'T_ENDWHILE' => 'endwhile',
|
||
|
'T_EVAL' => 'eval',
|
||
|
'T_EXIT' => 'exit call',
|
||
|
'T_EXTENDS' => 'extends',
|
||
|
'T_FILE' => '__FILE__',
|
||
|
'T_FINAL' => 'final',
|
||
|
'T_FINALLY' => 'finally',
|
||
|
'T_FOR' => 'for',
|
||
|
'T_FOREACH' => 'foreach',
|
||
|
'T_FUNCTION' => 'function',
|
||
|
'T_FUNC_C' => '__FUNCTION__',
|
||
|
'T_GLOBAL' => 'global',
|
||
|
'T_GOTO' => 'goto',
|
||
|
'T_HALT_COMPILER' => '__halt_compiler',
|
||
|
'T_IF' => 'if',
|
||
|
'T_IMPLEMENTS' => 'implements',
|
||
|
'T_INC' => '++ (increment)',
|
||
|
'T_INCLUDE' => 'include',
|
||
|
'T_INCLUDE_ONCE' => 'include_once',
|
||
|
'T_INSTANCEOF' => 'instanceof',
|
||
|
'T_INSTEADOF' => 'insteadof',
|
||
|
'T_INT_CAST' => 'int cast',
|
||
|
'T_INTERFACE' => 'interface',
|
||
|
'T_ISSET' => 'isset',
|
||
|
'T_IS_EQUAL' => "'=='",
|
||
|
'T_IS_GREATER_OR_EQUAL' => "'>='",
|
||
|
'T_IS_IDENTICAL' => "'==='",
|
||
|
'T_IS_NOT_EQUAL' => "'!=' or '<>'",
|
||
|
'T_IS_NOT_IDENTICAL' => "'!=='",
|
||
|
'T_IS_SMALLER_OR_EQUAL' => "'<='",
|
||
|
'T_LINE' => '__LINE__',
|
||
|
'T_LIST' => 'list',
|
||
|
'T_LNUMBER' => 'number',
|
||
|
'T_LOGICAL_AND' => "'and'",
|
||
|
'T_LOGICAL_OR' => "'or'",
|
||
|
'T_LOGICAL_XOR' => "'xor'",
|
||
|
'T_METHOD_C' => '__METHOD__',
|
||
|
'T_MINUS_EQUAL' => "'-='",
|
||
|
'T_MOD_EQUAL' => "'%='",
|
||
|
'T_MUL_EQUAL' => "'*='",
|
||
|
'T_NAMESPACE' => 'namespace',
|
||
|
'T_NEW' => 'new',
|
||
|
'T_NUM_STRING' => 'array index in a string',
|
||
|
'T_NS_C' => '__NAMESPACE__',
|
||
|
'T_NS_SEPARATOR' => 'namespace seperator',
|
||
|
'T_OBJECT_CAST' => 'object cast',
|
||
|
'T_OBJECT_OPERATOR' => "'->'",
|
||
|
'T_OLD_FUNCTION' => 'old_function',
|
||
|
'T_OPEN_TAG' => "'<?php' or '<?'",
|
||
|
'T_OPEN_TAG_WITH_ECHO' => "'<?php echo '",
|
||
|
'T_OR_EQUAL' => "'|='",
|
||
|
'T_PAAMAYIM_NEKUDOTAYIM' => "'::'",
|
||
|
'T_PLUS_EQUAL' => "'+='",
|
||
|
'T_PRINT' => 'print',
|
||
|
'T_PRIVATE' => 'private',
|
||
|
'T_PUBLIC' => 'public',
|
||
|
'T_PROTECTED' => 'protected',
|
||
|
'T_REQUIRE' => 'require',
|
||
|
'T_REQUIRE_ONCE' => 'require_once',
|
||
|
'T_RETURN' => 'return',
|
||
|
'T_SL' => "'<<'",
|
||
|
'T_SL_EQUAL' => "'<<='",
|
||
|
'T_SR' => "'>>'",
|
||
|
'T_SR_EQUAL' => "'>>='",
|
||
|
'T_START_HEREDOC' => "'<<<'",
|
||
|
'T_STATIC' => 'static',
|
||
|
'T_STRING' => 'string',
|
||
|
'T_STRING_CAST' => 'string cast',
|
||
|
'T_SWITCH' => 'switch',
|
||
|
'T_THROW' => 'throw',
|
||
|
'T_TRY' => 'try',
|
||
|
'T_TRAIT' => 'trait',
|
||
|
'T_TRAIT_C' => '__trait__',
|
||
|
'T_UNSET' => 'unset',
|
||
|
'T_UNSET_CAST' => 'unset cast',
|
||
|
'T_USE' => 'use',
|
||
|
'T_VAR' => 'var',
|
||
|
'T_VARIABLE' => 'variable',
|
||
|
'T_WHILE' => 'while',
|
||
|
'T_WHITESPACE' => 'whitespace',
|
||
|
'T_XOR_EQUAL' => "'^='",
|
||
|
'T_YIELD' => 'yield'
|
||
|
);
|
||
|
|
||
|
private static $syntaxMap = array(
|
||
|
'const' => 'syntax-literal',
|
||
|
'reference_ampersand' => 'syntax-function',
|
||
|
|
||
|
T_COMMENT => 'syntax-comment',
|
||
|
T_DOC_COMMENT => 'syntax-comment',
|
||
|
|
||
|
T_ABSTRACT => 'syntax-keyword',
|
||
|
T_AS => 'syntax-keyword',
|
||
|
T_BREAK => 'syntax-keyword',
|
||
|
T_CASE => 'syntax-keyword',
|
||
|
T_CATCH => 'syntax-keyword',
|
||
|
T_CLASS => 'syntax-keyword',
|
||
|
|
||
|
T_CONST => 'syntax-keyword',
|
||
|
|
||
|
T_CONTINUE => 'syntax-keyword',
|
||
|
T_DECLARE => 'syntax-keyword',
|
||
|
T_DEFAULT => 'syntax-keyword',
|
||
|
T_DO => 'syntax-keyword',
|
||
|
|
||
|
T_ELSE => 'syntax-keyword',
|
||
|
T_ELSEIF => 'syntax-keyword',
|
||
|
T_ENDDECLARE => 'syntax-keyword',
|
||
|
T_ENDFOR => 'syntax-keyword',
|
||
|
T_ENDFOREACH => 'syntax-keyword',
|
||
|
T_ENDIF => 'syntax-keyword',
|
||
|
T_ENDSWITCH => 'syntax-keyword',
|
||
|
T_ENDWHILE => 'syntax-keyword',
|
||
|
T_EXTENDS => 'syntax-keyword',
|
||
|
|
||
|
T_FINAL => 'syntax-keyword',
|
||
|
T_FINALLY => 'syntax-keyword',
|
||
|
T_FOR => 'syntax-keyword',
|
||
|
T_FOREACH => 'syntax-keyword',
|
||
|
T_FUNCTION => 'syntax-keyword',
|
||
|
T_GLOBAL => 'syntax-keyword',
|
||
|
T_GOTO => 'syntax-keyword',
|
||
|
|
||
|
T_IF => 'syntax-keyword',
|
||
|
T_IMPLEMENTS => 'syntax-keyword',
|
||
|
T_INSTANCEOF => 'syntax-keyword',
|
||
|
T_INSTEADOF => 'syntax-keyword',
|
||
|
T_INTERFACE => 'syntax-keyword',
|
||
|
|
||
|
T_LOGICAL_AND => 'syntax-keyword',
|
||
|
T_LOGICAL_OR => 'syntax-keyword',
|
||
|
T_LOGICAL_XOR => 'syntax-keyword',
|
||
|
T_NAMESPACE => 'syntax-keyword',
|
||
|
T_NEW => 'syntax-keyword',
|
||
|
T_PRIVATE => 'syntax-keyword',
|
||
|
T_PUBLIC => 'syntax-keyword',
|
||
|
T_PROTECTED => 'syntax-keyword',
|
||
|
T_RETURN => 'syntax-keyword',
|
||
|
T_STATIC => 'syntax-keyword',
|
||
|
T_SWITCH => 'syntax-keyword',
|
||
|
T_THROW => 'syntax-keyword',
|
||
|
T_TRAIT => 'syntax-keyword',
|
||
|
T_TRY => 'syntax-keyword',
|
||
|
T_USE => 'syntax-keyword',
|
||
|
T_VAR => 'syntax-keyword',
|
||
|
T_WHILE => 'syntax-keyword',
|
||
|
T_YIELD => 'syntax-keyword',
|
||
|
|
||
|
// __VAR__ type magic constants
|
||
|
T_CLASS_C => 'syntax-literal',
|
||
|
T_DIR => 'syntax-literal',
|
||
|
T_FILE => 'syntax-literal',
|
||
|
T_FUNC_C => 'syntax-literal',
|
||
|
T_LINE => 'syntax-literal',
|
||
|
T_METHOD_C => 'syntax-literal',
|
||
|
T_NS_C => 'syntax-literal',
|
||
|
T_TRAIT_C => 'syntax-literal',
|
||
|
|
||
|
T_DNUMBER => 'syntax-literal',
|
||
|
T_LNUMBER => 'syntax-literal',
|
||
|
|
||
|
T_CONSTANT_ENCAPSED_STRING => 'syntax-string',
|
||
|
T_VARIABLE => 'syntax-variable',
|
||
|
|
||
|
// this is for unescaped strings, which appear differently
|
||
|
// this includes function names
|
||
|
T_STRING => 'syntax-function',
|
||
|
|
||
|
// in build keywords, which work like functions
|
||
|
T_ARRAY => 'syntax-function',
|
||
|
T_CLONE => 'syntax-function',
|
||
|
T_ECHO => 'syntax-function',
|
||
|
T_EMPTY => 'syntax-function',
|
||
|
T_EVAL => 'syntax-function',
|
||
|
T_EXIT => 'syntax-function',
|
||
|
T_HALT_COMPILER => 'syntax-function',
|
||
|
T_INCLUDE => 'syntax-function',
|
||
|
T_INCLUDE_ONCE => 'syntax-function',
|
||
|
T_ISSET => 'syntax-function',
|
||
|
T_LIST => 'syntax-function',
|
||
|
T_REQUIRE_ONCE => 'syntax-function',
|
||
|
T_PRINT => 'syntax-function',
|
||
|
T_REQUIRE => 'syntax-function',
|
||
|
T_UNSET => 'syntax-function'
|
||
|
);
|
||
|
|
||
|
/**
|
||
|
* A list of methods which are known to call the autoloader,
|
||
|
* but should not error, if the class is not found.
|
||
|
*
|
||
|
* They are allowed to fail, so we don't store a class not
|
||
|
* found exception if they do.
|
||
|
*/
|
||
|
private static $SAFE_AUTOLOADER_FUNCTIONS = array(
|
||
|
'class_exists',
|
||
|
'interface_exists',
|
||
|
'method_exists',
|
||
|
'property_exists',
|
||
|
'is_subclass_of'
|
||
|
);
|
||
|
|
||
|
/**
|
||
|
* When returning values, if a mime type is set,
|
||
|
* then PHP Error should only output if the mime type
|
||
|
* is one of these.
|
||
|
*/
|
||
|
private static $ALLOWED_RETURN_MIME_TYPES = array(
|
||
|
'text/html',
|
||
|
'application/xhtml+xml'
|
||
|
);
|
||
|
|
||
|
private static function isIIS() {
|
||
|
return (
|
||
|
isset($_SERVER['SERVER_SOFTWARE']) &&
|
||
|
strpos($_SERVER['SERVER_SOFTWARE'], 'IIS/') !== false
|
||
|
) || (
|
||
|
isset($_SERVER['_FCGI_X_PIPE_']) &&
|
||
|
strpos($_SERVER['_FCGI_X_PIPE_'], 'IISFCGI') !== false
|
||
|
);
|
||
|
}
|
||
|
|
||
|
private static function isBinaryRequest() {
|
||
|
$response = ErrorHandler::getResponseHeaders();
|
||
|
|
||
|
foreach ( $response as $key => $value ) {
|
||
|
if ( strtolower($key) === 'content-transfer-encoding' ) {
|
||
|
return strtolower($value) === 'binary';
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* This attempts to state if this is *not* a PHP request,
|
||
|
* but it cannot say if it *is* a PHP request. It achieves
|
||
|
* this by looking for a mime type.
|
||
|
*
|
||
|
* For example if the mime type is JavaScript, then we
|
||
|
* know it's not PHP. However there is no "yes, this is
|
||
|
* definitely a normal HTML response" flag we can check.
|
||
|
*/
|
||
|
private static function isNonPHPRequest() {
|
||
|
/*
|
||
|
* Check if we are a mime type that isn't allowed.
|
||
|
*
|
||
|
* If an allowed type is found, then we return false,
|
||
|
* as were are a PHP Request.
|
||
|
*
|
||
|
* Anything else found, returns true, as that means
|
||
|
* we are dealing with something unknown.
|
||
|
*/
|
||
|
$response = ErrorHandler::getResponseHeaders();
|
||
|
|
||
|
foreach ( $response as $key => $value ) {
|
||
|
if ( strtolower($key) === 'content-type' ) {
|
||
|
foreach ( ErrorHandler::$ALLOWED_RETURN_MIME_TYPES as $type ) {
|
||
|
if ( stripos($value, $type) !== false ) {
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Looks up a description for the symbol given,
|
||
|
* and if found, it is returned.
|
||
|
*
|
||
|
* If it's not found, then the symbol given is returned.
|
||
|
*/
|
||
|
private static function phpSymbolToDescription( $symbol ) {
|
||
|
if ( isset(ErrorHandler::$PHP_SYMBOL_MAPPINGS[$symbol]) ) {
|
||
|
return ErrorHandler::$PHP_SYMBOL_MAPPINGS[$symbol];
|
||
|
} else {
|
||
|
return "'$symbol'";
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Attempts to syntax highlight the code snippet done.
|
||
|
*
|
||
|
* This is then returned as HTML, ready to be dumped to the screen.
|
||
|
*
|
||
|
* @param code An array of code lines to syntax highlight.
|
||
|
* @return HTML version of the code given, syntax highlighted.
|
||
|
*/
|
||
|
private static function syntaxHighlight( $code ) {
|
||
|
$syntaxMap = ErrorHandler::$syntaxMap;
|
||
|
|
||
|
// @supress invalid code raises a warning
|
||
|
$tokens = @token_get_all( "<?php " . $code . " ?" . ">" );
|
||
|
$html = array();
|
||
|
$len = count($tokens)-1;
|
||
|
$inString = false;
|
||
|
$stringBuff = null;
|
||
|
$skip = false;
|
||
|
|
||
|
for ( $i = 1; $i < $len; $i++ ) {
|
||
|
$token = $tokens[$i];
|
||
|
|
||
|
if ( is_array($token) ) {
|
||
|
$type = $token[0];
|
||
|
$code = $token[1];
|
||
|
} else {
|
||
|
$type = null;
|
||
|
$code = $token;
|
||
|
}
|
||
|
|
||
|
// work out any whitespace padding
|
||
|
if ( strpos($code, "\n") !== false && trim($code) === '' ) {
|
||
|
if ( $inString ) {
|
||
|
$html[]= "<span class='syntax-string'>" . join('', $stringBuff);
|
||
|
$stringBuff = array();
|
||
|
}
|
||
|
} else if ( $code === '&' ) {
|
||
|
if ( $i < $len ) {
|
||
|
$next = $tokens[$i+1];
|
||
|
|
||
|
if ( is_array($next) && $next[0] === T_VARIABLE ) {
|
||
|
$type = 'reference_ampersand';
|
||
|
}
|
||
|
}
|
||
|
} else if ( $code === '"' || $code === "'" ) {
|
||
|
if ( $inString ) {
|
||
|
$html[]= "<span class='syntax-string'>" . join('', $stringBuff) . htmlspecialchars($code) . "</span>";
|
||
|
$stringBuff = null;
|
||
|
$skip = true;
|
||
|
} else {
|
||
|
$stringBuff = array();
|
||
|
}
|
||
|
|
||
|
$inString = !$inString;
|
||
|
} else if ( $type === T_STRING ) {
|
||
|
$matches = array();
|
||
|
preg_match(ErrorHandler::REGEX_PHP_CONST_IDENTIFIER, $code, $matches);
|
||
|
|
||
|
if ( $matches && strlen($matches[0]) === strlen($code) ) {
|
||
|
$type = 'const';
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( $skip ) {
|
||
|
$skip = false;
|
||
|
} else {
|
||
|
$code = htmlspecialchars( $code );
|
||
|
|
||
|
if ( $type !== null && isset($syntaxMap[$type]) ) {
|
||
|
$class = $syntaxMap[$type];
|
||
|
|
||
|
if ( $type === T_CONSTANT_ENCAPSED_STRING && strpos($code, "\n") !== false ) {
|
||
|
$append = "<span class='$class'>" .
|
||
|
join(
|
||
|
"</span>\n<span class='$class'>",
|
||
|
explode( "\n", $code )
|
||
|
) .
|
||
|
"</span>" ;
|
||
|
} else if ( strrpos($code, "\n") === strlen($code)-1 ) {
|
||
|
$append = "<span class='$class'>" . substr($code, 0, strlen($code)-1) . "</span>\n";
|
||
|
} else {
|
||
|
$append = "<span class='$class'>$code</span>";
|
||
|
}
|
||
|
} else if ( $inString && $code !== '"' ) {
|
||
|
$append = "<span class='syntax-string'>$code</span>";
|
||
|
} else {
|
||
|
$append = $code;
|
||
|
}
|
||
|
|
||
|
if ( $inString ) {
|
||
|
$stringBuff[]= $append;
|
||
|
} else {
|
||
|
$html[]= $append;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( $stringBuff !== null ) {
|
||
|
$html[]= "<span class='syntax-string'>" . join('', $stringBuff) . '</span>';
|
||
|
$stringBuff = null;
|
||
|
}
|
||
|
|
||
|
return join( '', $html );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Splits a given function name into it's 'class, function' parts.
|
||
|
* If there is no class, then null is returned.
|
||
|
*
|
||
|
* It also returns these parts in an array of: array( $className, $functionName );
|
||
|
*
|
||
|
* Usage:
|
||
|
*
|
||
|
* list( $class, $function ) = ErrorHandler::splitFunction( $name );
|
||
|
*
|
||
|
* @param name The function name to split.
|
||
|
* @return An array containing class and function name.
|
||
|
*/
|
||
|
private static function splitFunction( $name ) {
|
||
|
$name = preg_replace( '/\\(\\)$/', '', $name );
|
||
|
|
||
|
if ( strpos($name, '::') !== false ) {
|
||
|
$parts = explode( '::', $name );
|
||
|
$className = $parts[0];
|
||
|
$type = '::';
|
||
|
$functionName = $parts[1];
|
||
|
} else if ( strpos($name, '->') !== false ) {
|
||
|
$parts = explode( '->', $name );
|
||
|
$className = $parts[0];
|
||
|
$type = '->';
|
||
|
$functionName = $parts[1];
|
||
|
} else {
|
||
|
$className = null;
|
||
|
$type = null;
|
||
|
$functionName = $name;
|
||
|
}
|
||
|
|
||
|
return array( $className, $type, $functionName );
|
||
|
}
|
||
|
|
||
|
private static function newArgument( $name, $type=false, $isPassedByReference=false, $isOptional=false, $optionalValue=null, $highlight=false ) {
|
||
|
if ( $name instanceof ReflectionParameter ) {
|
||
|
$highlight = func_num_args() > 1 ?
|
||
|
$highlight = $type :
|
||
|
false;
|
||
|
|
||
|
$klass = $name->getDeclaringClass();
|
||
|
$functionName = $name->getDeclaringFunction()->name;
|
||
|
if ( $klass !== null ) {
|
||
|
$klass = $klass->name;
|
||
|
}
|
||
|
|
||
|
$export = ReflectionParameter::export(
|
||
|
( $klass ?
|
||
|
array( "\\$klass", $functionName ) :
|
||
|
$functionName ),
|
||
|
$name->name,
|
||
|
true
|
||
|
);
|
||
|
|
||
|
$paramType = preg_replace('/.*?(\w+)\s+\$'.$name->name.'.*/', '\\1', $export);
|
||
|
if ( strpos($paramType, '[') !== false || strlen($paramType) === 0 ) {
|
||
|
$paramType = null;
|
||
|
}
|
||
|
|
||
|
return ErrorHandler::newArgument(
|
||
|
$name->name,
|
||
|
$paramType,
|
||
|
$name->isPassedByReference(),
|
||
|
$name->isDefaultValueAvailable(),
|
||
|
( $name->isDefaultValueAvailable() ?
|
||
|
var_export( $name->getDefaultValue(), true ) :
|
||
|
null ),
|
||
|
( func_num_args() > 1 ?
|
||
|
$type :
|
||
|
false )
|
||
|
);
|
||
|
} else {
|
||
|
return array(
|
||
|
'name' => $name,
|
||
|
'has_type' => ( $type !== false ),
|
||
|
'type' => $type,
|
||
|
'is_reference' => $isPassedByReference,
|
||
|
'has_default' => $isOptional,
|
||
|
'default_val' => $optionalValue,
|
||
|
'is_highlighted' => $highlight
|
||
|
);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private static function syntaxHighlightFunctionMatch( $match, &$stackTrace, $highlightArg=null, &$numHighlighted=0 ) {
|
||
|
list( $className, $type, $functionName ) = ErrorHandler::splitFunction( $match );
|
||
|
|
||
|
// is class::method()
|
||
|
if ( $className !== null ) {
|
||
|
$reflectFun = new ReflectionMethod( $className, $functionName );
|
||
|
// is a function
|
||
|
} else if ( $functionName === '{closure}' ) {
|
||
|
return '<span class="syntax-variable">$closure</span>';
|
||
|
} else {
|
||
|
$reflectFun = new ReflectionFunction( $functionName );
|
||
|
}
|
||
|
|
||
|
if ( $reflectFun ) {
|
||
|
$params = $reflectFun->getParameters();
|
||
|
|
||
|
if ( $params ) {
|
||
|
$args = array();
|
||
|
$min = 0;
|
||
|
foreach( $params as $i => $param ) {
|
||
|
$arg = ErrorHandler::newArgument( $param );
|
||
|
|
||
|
if ( ! $arg['has_default'] ) {
|
||
|
$min = $i;
|
||
|
}
|
||
|
|
||
|
$args[]= $arg;
|
||
|
}
|
||
|
|
||
|
if ( $highlightArg !== null ) {
|
||
|
for ( $i = $highlightArg; $i <= $min; $i++ ) {
|
||
|
$args[$i]['is_highlighted'] = true;
|
||
|
}
|
||
|
|
||
|
$numHighlighted = $min-$highlightArg;
|
||
|
}
|
||
|
|
||
|
if ( $className !== null ) {
|
||
|
if ( $stackTrace && isset($stackTrace[1]) && isset($stackTrace[1]['type']) ) {
|
||
|
$type = htmlspecialchars( $stackTrace[1]['type'] );
|
||
|
}
|
||
|
} else {
|
||
|
$type = null;
|
||
|
}
|
||
|
|
||
|
return ErrorHandler::syntaxHighlightFunction( $className, $type, $functionName, $args );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return null;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns the values given, as HTML, syntax highlighted.
|
||
|
* It's a shorter, slightly faster, more no-nonsense approach
|
||
|
* then 'syntaxHighlight'.
|
||
|
*
|
||
|
* This is for syntax highlighting:
|
||
|
* - fun( [args] )
|
||
|
* - class->fun( [args] )
|
||
|
* - class::fun( [args] )
|
||
|
*
|
||
|
* Class and type can be null, to denote no class, but are not optional.
|
||
|
*/
|
||
|
private static function syntaxHighlightFunction( $class, $type, $fun, &$args=null ) {
|
||
|
$info = array();
|
||
|
|
||
|
// set the info
|
||
|
if ( isset($class) && $class && isset($type) && $type ) {
|
||
|
if ( $type === '->' ) {
|
||
|
$type = '->';
|
||
|
}
|
||
|
|
||
|
$info []= "<span class='syntax-class'>$class</span>$type";
|
||
|
}
|
||
|
|
||
|
if ( isset($fun) && $fun ) {
|
||
|
$info []= "<span class='syntax-function'>$fun</span>";
|
||
|
}
|
||
|
|
||
|
if ( $args ) {
|
||
|
$info []= '( ';
|
||
|
|
||
|
foreach ($args as $i => $arg) {
|
||
|
if ( $i > 0 ) {
|
||
|
$info[]= ', ';
|
||
|
}
|
||
|
|
||
|
if ( is_string($arg) ) {
|
||
|
$info[]= $arg;
|
||
|
} else {
|
||
|
$highlight = $arg['is_highlighted'];
|
||
|
$name = $arg['name'];
|
||
|
|
||
|
if ( $highlight ) {
|
||
|
$info[]= '<span class="syntax-higlight-variable">';
|
||
|
}
|
||
|
|
||
|
if ( $name === '_' ) {
|
||
|
$info[]= '<span class="syntax-variable-not-important">';
|
||
|
}
|
||
|
|
||
|
if ( $arg['has_type'] ) {
|
||
|
$info []= "<span class='syntax-class'>";
|
||
|
$info []= $arg['type'];
|
||
|
$info []= '</span> ';
|
||
|
}
|
||
|
|
||
|
if ( $arg['is_reference'] ) {
|
||
|
$info []= '<span class="syntax-function">&</span>';
|
||
|
}
|
||
|
|
||
|
$info []= "<span class='syntax-variable'>\$$name</span>";
|
||
|
|
||
|
if ( $arg['has_default'] ) {
|
||
|
$info []= '=<span class="syntax-literal">' . $arg['default_val'] . '</span>';
|
||
|
}
|
||
|
|
||
|
if ( $name === '_' ) {
|
||
|
$info[]= '</span>';
|
||
|
}
|
||
|
if ( $highlight ) {
|
||
|
$info[]= '</span>';
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$info []= ' )';
|
||
|
} else {
|
||
|
$info []= '()';
|
||
|
}
|
||
|
|
||
|
return join( '', $info );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Checks if the item is in options, and if it is, then it is removed and returned.
|
||
|
*
|
||
|
* If it is not found, or if options is not an array, then the alt is returned.
|
||
|
*/
|
||
|
private static function optionsPop( &$options, $key, $alt=null ) {
|
||
|
if ( $options && isset($options[$key]) ) {
|
||
|
$val = $options[$key];
|
||
|
unset( $options[$key] );
|
||
|
|
||
|
return $val;
|
||
|
} else {
|
||
|
$iniAlt = @get_cfg_var( ErrorHandler::PHP_ERROR_INI_PREFIX . '.' . $key );
|
||
|
|
||
|
if ( $iniAlt !== false ) {
|
||
|
return $iniAlt;
|
||
|
} else {
|
||
|
return $alt;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private static function folderTypeToCSS( $type ) {
|
||
|
if ( $type === ErrorHandler::FILE_TYPE_ROOT ) {
|
||
|
return 'file-root';
|
||
|
} else if ( $type === ErrorHandler::FILE_TYPE_IGNORE ) {
|
||
|
return 'file-ignore';
|
||
|
} else if ( $type === ErrorHandler::FILE_TYPE_APPLICATION ) {
|
||
|
return 'file-app';
|
||
|
} else {
|
||
|
return 'file-common';
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private static function isFolderType( &$folders, $longest, $file ) {
|
||
|
$parts = explode( '/', $file );
|
||
|
|
||
|
$len = min( count($parts), $longest );
|
||
|
|
||
|
for ( $i = $len; $i > 0; $i-- ) {
|
||
|
if ( isset($folders[$i]) ) {
|
||
|
$folderParts = &$folders[ $i ];
|
||
|
|
||
|
$success = false;
|
||
|
for ( $j = 0; $j < count($folderParts); $j++ ) {
|
||
|
$folderNames = $folderParts[$j];
|
||
|
|
||
|
for ( $k = 0; $k < count($folderNames); $k++ ) {
|
||
|
if ( $folderNames[$k] === $parts[$k] ) {
|
||
|
$success = true;
|
||
|
} else {
|
||
|
$success = false;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( $success ) {
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
private static function setFolders( &$origFolders, &$longest, $folders ) {
|
||
|
$newFolders = array();
|
||
|
$newLongest = 0;
|
||
|
|
||
|
if ( $folders ) {
|
||
|
if ( is_array($folders) ) {
|
||
|
foreach ( $folders as $folder ) {
|
||
|
ErrorHandler::setFoldersInner( $newFolders, $newLongest, $folder );
|
||
|
}
|
||
|
} else if ( is_string($folders) ) {
|
||
|
ErrorHandler::setFoldersInner( $newFolders, $newLongest, $folders );
|
||
|
} else {
|
||
|
throw new Exception( "Unknown value given for folder: " . $folders );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$origFolders = $newFolders;
|
||
|
$longest = $newLongest;
|
||
|
}
|
||
|
|
||
|
private static function setFoldersInner( &$newFolders, &$newLongest, $folder ) {
|
||
|
$folder = str_replace( '\\', '/', $folder );
|
||
|
$folder = preg_replace( '/(^\\/+)|(\\/+$)/', '', $folder );
|
||
|
$parts = explode( '/', $folder );
|
||
|
$count = count( $parts );
|
||
|
|
||
|
$newLongest = max( $newLongest, $count );
|
||
|
|
||
|
if ( isset($newFolders[$count]) ) {
|
||
|
$folds = &$newFolders[$count];
|
||
|
$folds[]= $parts;
|
||
|
} else {
|
||
|
$newFolders[$count] = array( $parts );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private static function getRequestHeaders() {
|
||
|
if ( function_exists('getallheaders') ) {
|
||
|
return getallheaders();
|
||
|
} else {
|
||
|
$headers = array();
|
||
|
|
||
|
foreach ( $_SERVER as $key => $value ) {
|
||
|
if ( strpos($key, 'HTTP_') === 0 ) {
|
||
|
$key = str_replace( " ", "-", ucwords(strtolower( str_replace("_", " ", substr($key, 5)) )) );
|
||
|
$headers[ $key ] = $value;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $headers;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private static function getResponseHeaders() {
|
||
|
$headers = function_exists('apache_response_headers') ?
|
||
|
apache_response_headers() :
|
||
|
array() ;
|
||
|
|
||
|
/*
|
||
|
* Merge the headers_list into apache_response_headers.
|
||
|
*
|
||
|
* This is because sometimes things are in one, which are
|
||
|
* not present in the other.
|
||
|
*/
|
||
|
if ( function_exists('headers_list') ) {
|
||
|
$hList = headers_list();
|
||
|
|
||
|
foreach ($hList as $header) {
|
||
|
$header = explode(":", $header);
|
||
|
$headers[ array_shift($header) ] = trim( implode(":", $header) );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $headers;
|
||
|
}
|
||
|
|
||
|
public static function identifyTypeHTML( $arg, $recurseLevels=1 ) {
|
||
|
if ( ! is_array($arg) && !is_object($arg) ) {
|
||
|
if ( is_string($arg) ) {
|
||
|
return "<span class='syntax-string'>"" . htmlentities($arg) . ""</span>";
|
||
|
} else {
|
||
|
return "<span class='syntax-literal'>" . var_export( $arg, true ) . '</span>';
|
||
|
}
|
||
|
} else if ( is_array($arg) ) {
|
||
|
if ( count($arg) === 0 ) {
|
||
|
return "[]";
|
||
|
} else if ( $recurseLevels > 0 ) {
|
||
|
$argArr = array();
|
||
|
|
||
|
foreach ($arg as $ag) {
|
||
|
$argArr[]= ErrorHandler::identifyTypeHTML( $ag, $recurseLevels-1 );
|
||
|
}
|
||
|
|
||
|
if ( ($recurseLevels % 2) === 0 ) {
|
||
|
return "[" . join(', ', $argArr) . "]";
|
||
|
} else {
|
||
|
return "[ " . join(', ', $argArr) . " ]";
|
||
|
}
|
||
|
} else {
|
||
|
return "[...]";
|
||
|
}
|
||
|
} else if ( get_class($arg) === 'Closure' ) {
|
||
|
return '<span class="syntax-variable">$Closure</span>()';
|
||
|
} else {
|
||
|
$argKlass = get_class( $arg );
|
||
|
|
||
|
if ( preg_match(ErrorHandler::REGEX_PHP_CONST_IDENTIFIER, $argKlass) ) {
|
||
|
return '<span class="syntax-literal">$' . $argKlass . '</span>';
|
||
|
} else {
|
||
|
return '<span class="syntax-variable">$' . $argKlass . '</span>';
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private $saveUrl;
|
||
|
private $isSavingEnabled;
|
||
|
|
||
|
private $cachedFiles;
|
||
|
|
||
|
private $isShutdownRegistered;
|
||
|
private $isOn;
|
||
|
|
||
|
private $ignoreFolders = array();
|
||
|
private $ignoreFoldersLongest = 0;
|
||
|
|
||
|
private $applicationFolders = array();
|
||
|
private $applicationFoldersLongest = 0;
|
||
|
|
||
|
private $defaultErrorReportingOn;
|
||
|
private $defaultErrorReportingOff;
|
||
|
private $applicationRoot;
|
||
|
private $serverName;
|
||
|
|
||
|
private $catchClassNotFound;
|
||
|
private $catchSurpressedErrors;
|
||
|
private $catchAjaxErrors;
|
||
|
|
||
|
private $backgroundText;
|
||
|
private $numLines;
|
||
|
|
||
|
private $displayLineNumber;
|
||
|
private $htmlOnly;
|
||
|
|
||
|
private $isBufferSetup;
|
||
|
private $bufferOutputStr;
|
||
|
private $bufferOutput;
|
||
|
|
||
|
private $isAjax;
|
||
|
|
||
|
private $lastGlobalErrorHandler;
|
||
|
|
||
|
private $classNotFoundException;
|
||
|
|
||
|
/**
|
||
|
* = Options =
|
||
|
*
|
||
|
* All options are optional, and so is passing in an options item.
|
||
|
* You don't have to supply any, it's up to you.
|
||
|
*
|
||
|
* Note that if 'php_error.force_disable' is true, then this object
|
||
|
* will try to look like it works, but won't actually do anything.
|
||
|
*
|
||
|
* All options can also be passed in from 'php.ini'. You do this
|
||
|
* by setting it with 'php_error.' prefix. For example:
|
||
|
*
|
||
|
* php_error.catch_ajax_errors = On
|
||
|
* php_error.error_reporting_on = E_ALL | E_STRICT
|
||
|
*
|
||
|
* Includes:
|
||
|
* = Types of errors this will catch =
|
||
|
* - catch_ajax_errors When on, this will inject JS Ajax wrapping code, to allow this to catch any future JSON errors. Defaults to true.
|
||
|
* - catch_supressed_errors The @ supresses errors. If set to true, then they are still reported anyway, but respected when false. Defaults to false.
|
||
|
* - catch_class_not_found When true, loading a class that does not exist will be caught. This defaults to true.
|
||
|
*
|
||
|
* = Error reporting level =
|
||
|
* - error_reporting_on value for when errors are on, defaults to all errors
|
||
|
* - error_reporting_off value for when errors are off, defaults to php.ini's error_reporting.
|
||
|
*
|
||
|
* = Setup Details =
|
||
|
* - application_root When it's working out hte stack trace, this is the root folder of the application, to use as it's base.
|
||
|
* Defaults to the servers root directory.
|
||
|
*
|
||
|
* A relative path can be given, but lets be honest, an explicit path is the way to guarantee that you
|
||
|
* will get the path you want. My relative might not be the same as your relative.
|
||
|
*
|
||
|
* - snippet_num_lines The number of lines to display in the code snippet.
|
||
|
* That includes the line being reported.
|
||
|
*
|
||
|
* - server_name The name for this server, defaults to "$_SERVER['SERVER_NAME']"
|
||
|
*
|
||
|
* - ignore_folders This is allows you to highlight non-framework code in a stack trace.
|
||
|
* An array of folders to ignore, when working out the stack trace.
|
||
|
* This is folder prefixes in relation to the application_root, whatever that might be.
|
||
|
* They are only ignored if there is a file found outside of them.
|
||
|
* If you still don't get what this does, don't worry, it's here cos I use it.
|
||
|
*
|
||
|
* - application_folders Just like ignore, but anything found in these folders takes precedence
|
||
|
* over anything else.
|
||
|
*
|
||
|
* - background_text The text that appeares in the background. By default this is blank.
|
||
|
* Why? You can replace this with the name of your framework, for extra customization spice.
|
||
|
*
|
||
|
* - html_only By default, PHP Error only runs on ajax and HTML pages.
|
||
|
* If this is false, then it will also run when on non-HTML
|
||
|
* pages too, such as replying with images of JavaScript
|
||
|
* from your PHP. Defaults to true.
|
||
|
*
|
||
|
* - file_link When true, files are linked to from the CSS Stack trace, allowing you to open them.
|
||
|
* Defaults to true.
|
||
|
*
|
||
|
* - save_url The url of where to send files, to be saved.
|
||
|
* Note that 'enable_saving' must be on for this to be used (which it is by default).
|
||
|
*
|
||
|
* - enable_saving Can be true or false. When true, saving files is enabled, and when false, it is disabled.
|
||
|
* Defaults to true!
|
||
|
*
|
||
|
* @param options Optional, an array of values to customize this handler.
|
||
|
* @throws Exception This is raised if given an options that does *not* exist (so you know that option is meaningless).
|
||
|
*/
|
||
|
public function __construct( $options=null ) {
|
||
|
// there can only be one to rule them all
|
||
|
global $_php_error_global_handler;
|
||
|
if ( $_php_error_global_handler !== null ) {
|
||
|
$this->lastGlobalErrorHandler = $_php_error_global_handler;
|
||
|
} else {
|
||
|
$this->lastGlobalErrorHandler = null;
|
||
|
}
|
||
|
$_php_error_global_handler = $this;
|
||
|
|
||
|
$this->cachedFiles = array();
|
||
|
|
||
|
$this->isShutdownRegistered = false;
|
||
|
$this->isOn = false;
|
||
|
|
||
|
/*
|
||
|
* Deal with the options.
|
||
|
*
|
||
|
* They are removed one by one, and any left, will raise an error.
|
||
|
*/
|
||
|
|
||
|
$ignoreFolders = ErrorHandler::optionsPop( $options, 'ignore_folders' , null );
|
||
|
$appFolders = ErrorHandler::optionsPop( $options, 'application_folders', null );
|
||
|
|
||
|
if ( $ignoreFolders !== null ) {
|
||
|
ErrorHandler::setFolders( $this->ignoreFolders, $this->ignoreFoldersLongest, $ignoreFolders );
|
||
|
}
|
||
|
if ( $appFolders !== null ) {
|
||
|
ErrorHandler::setFolders( $this->applicationFolders, $this->applicationFoldersLongest, $appFolders );
|
||
|
}
|
||
|
|
||
|
$this->saveUrl = ErrorHandler::optionsPop( $options, 'save_url', $_SERVER['REQUEST_URI'] );
|
||
|
$this->isSavingEnabled = ErrorHandler::optionsPop( $options, 'enable_saving', true );
|
||
|
|
||
|
$this->defaultErrorReportingOn = ErrorHandler::optionsPop( $options, 'error_reporting_on' , -1 );
|
||
|
$this->defaultErrorReportingOff = ErrorHandler::optionsPop( $options, 'error_reporting_off' , error_reporting() );
|
||
|
|
||
|
$this->applicationRoot = ErrorHandler::optionsPop( $options, 'application_root' , $_SERVER['DOCUMENT_ROOT'] );
|
||
|
$this->serverName = ErrorHandler::optionsPop( $options, 'server_name' , $_SERVER['SERVER_NAME'] );
|
||
|
|
||
|
/*
|
||
|
* Relative paths might be given for document root,
|
||
|
* so we make it explicit.
|
||
|
*/
|
||
|
$dir = @realpath( $this->applicationRoot );
|
||
|
if ( ! is_string($dir) ) {
|
||
|
throw new Exception("Document root not found: " . $this->applicationRoot);
|
||
|
} else {
|
||
|
$this->applicationRoot = str_replace( '\\', '/', $dir );
|
||
|
}
|
||
|
|
||
|
$this->catchClassNotFound = !! ErrorHandler::optionsPop( $options, 'catch_class_not_found' , true );
|
||
|
$this->catchSurpressedErrors = !! ErrorHandler::optionsPop( $options, 'catch_supressed_errors', false );
|
||
|
$this->catchAjaxErrors = !! ErrorHandler::optionsPop( $options, 'catch_ajax_errors' , true );
|
||
|
|
||
|
$this->backgroundText = ErrorHandler::optionsPop( $options, 'background_text' , '' );
|
||
|
$this->numLines = ErrorHandler::optionsPop( $options, 'snippet_num_lines' , ErrorHandler::NUM_FILE_LINES );
|
||
|
$this->displayLineNumber = ErrorHandler::optionsPop( $options, 'display_line_numbers' , true );
|
||
|
|
||
|
$this->htmlOnly = !! ErrorHandler::optionsPop( $options, 'html_only', true );
|
||
|
|
||
|
$this->classNotFoundException = null;
|
||
|
|
||
|
$wordpress = ErrorHandler::optionsPop( $options, 'wordpress', false );
|
||
|
if ( $wordpress ) {
|
||
|
// php doesn't like | in constants and privates, so just set it directly : (
|
||
|
$this->defaultErrorReportingOn = E_ERROR | E_WARNING | E_PARSE | E_USER_DEPRECATED & ~E_DEPRECATED & ~E_STRICT;
|
||
|
}
|
||
|
|
||
|
$concrete5 = ErrorHandler::optionsPop( $options, 'concrete5', false );
|
||
|
if ( $concrete5 ) {
|
||
|
$this->defaultErrorReportingOn = E_ALL & ~E_NOTICE & ~E_STRICT & ~E_DEPRECATED;
|
||
|
}
|
||
|
|
||
|
if ( $options ) {
|
||
|
foreach ( $options as $key => $val ) {
|
||
|
throw new InvalidArgumentException( "Unknown option given $key" );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$this->isAjax = (
|
||
|
isset( $_SERVER['HTTP_X_REQUESTED_WITH'] ) &&
|
||
|
( $_SERVER['HTTP_X_REQUESTED_WITH'] === 'XMLHttpRequest' )
|
||
|
) || (
|
||
|
isset( $_REQUEST['php_error_is_ajax'] )
|
||
|
);
|
||
|
|
||
|
$this->isBufferSetup = false;
|
||
|
$this->bufferOutputStr = '';
|
||
|
$this->bufferOutput = false;
|
||
|
|
||
|
$this->startBuffer();
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- ---
|
||
|
* Public Functions
|
||
|
* --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- ---
|
||
|
*/
|
||
|
|
||
|
/**
|
||
|
* @return true if this is currently on, false if not.
|
||
|
*/
|
||
|
public function isOn() {
|
||
|
return $this->isOn;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @return If this is off, this returns true, otherwise false.
|
||
|
*/
|
||
|
public function isOff() {
|
||
|
return !$this->isOn;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Turns error reporting on.
|
||
|
*
|
||
|
* This will use the strictest error reporting available, or the
|
||
|
* level you pass in when creating this using the 'error_reporting_on'
|
||
|
* option.
|
||
|
*
|
||
|
* @return This error reporting handler, for method chaining.
|
||
|
*/
|
||
|
public function turnOn() {
|
||
|
$this->propagateTurnOff();
|
||
|
$this->setEnabled( true );
|
||
|
|
||
|
/*
|
||
|
* Check if file changes have been uploaded,
|
||
|
* and if so, save them.
|
||
|
*/
|
||
|
global $_php_error_is_ini_enabled;
|
||
|
if ( $_php_error_is_ini_enabled ) {
|
||
|
if ( $this->isSavingEnabled ) {
|
||
|
$headers = ErrorHandler::getRequestHeaders();
|
||
|
|
||
|
if ( isset($headers[ErrorHandler::HEADER_SAVE_FILE]) ) {
|
||
|
if ( isset($_POST) && isset($_POST[ErrorHandler::POST_FILE_LOCATION]) ) {
|
||
|
$files = $_POST[ErrorHandler::POST_FILE_LOCATION];
|
||
|
|
||
|
foreach ( $files as $file => $content ) {
|
||
|
@file_put_contents( $file, stripcslashes($content) );
|
||
|
}
|
||
|
|
||
|
exit(0);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $this;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Turns error reporting off.
|
||
|
*
|
||
|
* This will use the 'php.ini' setting for the error_reporting level,
|
||
|
* or one you have passed in if you used the 'error_reporting_off'
|
||
|
* option when creating this.
|
||
|
*
|
||
|
* @return This error reporting handler, for method chaining.
|
||
|
*/
|
||
|
public function turnOff() {
|
||
|
$this->setEnabled( false );
|
||
|
|
||
|
return $this;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Allows you to run a callback with strict errors turned off.
|
||
|
* Standard errors still apply, but this will use the default
|
||
|
* error and exception handlers.
|
||
|
*
|
||
|
* This is useful for when loading libraries which do not
|
||
|
* adhere to strict errors, such as Wordpress.
|
||
|
*
|
||
|
* To use:
|
||
|
*
|
||
|
* withoutErrors( function() {
|
||
|
* // unsafe code here
|
||
|
* });
|
||
|
*
|
||
|
* This will use the error_reporting value for when this is
|
||
|
* turned off.
|
||
|
*
|
||
|
* @param callback A PHP function to call.
|
||
|
* @return The result of calling the callback.
|
||
|
*/
|
||
|
public function withoutErrors( $callback ) {
|
||
|
if ( ! is_callable($callback) ) {
|
||
|
throw new Exception( "non callable callback given" );
|
||
|
}
|
||
|
|
||
|
if ( $this->isOn() ) {
|
||
|
$this->turnOff();
|
||
|
$result = $callback();
|
||
|
$this->turnOn();
|
||
|
|
||
|
return $result;
|
||
|
} else {
|
||
|
return $callback();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* This is the shutdown function, which should *only* be called
|
||
|
* via 'register_shutdown_function'.
|
||
|
*
|
||
|
* It's exposed because it has to be exposed.
|
||
|
*/
|
||
|
public function __onShutdown() {
|
||
|
global $_php_error_is_ini_enabled;
|
||
|
|
||
|
if ( $_php_error_is_ini_enabled ) {
|
||
|
if ( $this->isOn() ) {
|
||
|
$error = error_get_last();
|
||
|
|
||
|
// fatal and syntax errors
|
||
|
if (
|
||
|
$error && (
|
||
|
$error['type'] === 1 ||
|
||
|
$error['type'] === 4 ||
|
||
|
$error['type'] === 64
|
||
|
)
|
||
|
) {
|
||
|
$this->reportError( $error['type'], $error['message'], $error['line'], $error['file'] );
|
||
|
} else {
|
||
|
$this->endBuffer();
|
||
|
}
|
||
|
} else {
|
||
|
$this->endBuffer();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- ---
|
||
|
* Private Functions
|
||
|
* --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- --- ---
|
||
|
*/
|
||
|
|
||
|
private function propagateTurnOff() {
|
||
|
if ( $this->lastGlobalErrorHandler !== null ) {
|
||
|
$this->lastGlobalErrorHandler->turnOff();
|
||
|
$this->lastGlobalErrorHandler->propagateTurnOff();
|
||
|
$this->lastGlobalErrorHandler = null;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* This is intended to be used closely with 'onShutdown'.
|
||
|
* It ensures that output buffering is turned on.
|
||
|
*
|
||
|
* Why? The user may output content, and *then* hit an error.
|
||
|
* We cannot replace the page if this happens,
|
||
|
* because they have already outputted information.
|
||
|
*
|
||
|
* So we buffer the page, and then output at the end of the page,
|
||
|
* or when an error strikes.
|
||
|
*/
|
||
|
private function startBuffer() {
|
||
|
global $_php_error_is_ini_enabled;
|
||
|
|
||
|
if ( $_php_error_is_ini_enabled && !$this->isBufferSetup ) {
|
||
|
$this->isBufferSetup = true;
|
||
|
|
||
|
ini_set( 'implicit_flush', false );
|
||
|
ob_implicit_flush( false );
|
||
|
|
||
|
if ( ! @ini_get('output_buffering') ) {
|
||
|
@ini_set( 'output_buffering', 'on' );
|
||
|
}
|
||
|
|
||
|
$output = '';
|
||
|
$bufferOutput = true;
|
||
|
|
||
|
$this->bufferOutputStr = &$output;
|
||
|
$this->bufferOutput = &$bufferOutput;
|
||
|
|
||
|
ob_start( function($string) use (&$output, &$bufferOutput) {
|
||
|
if ( $bufferOutput ) {
|
||
|
$output .= $string;
|
||
|
return '';
|
||
|
} else {
|
||
|
$temp = $output . $string;
|
||
|
$output = '';
|
||
|
return $temp;
|
||
|
}
|
||
|
});
|
||
|
|
||
|
$self = $this;
|
||
|
register_shutdown_function( function() use ( $self ) {
|
||
|
$self->__onShutdown();
|
||
|
});
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Turns off buffering, and discards anything buffered
|
||
|
* so far.
|
||
|
*
|
||
|
* This will return what has been buffered incase you
|
||
|
* do want it. However otherwise, it will be lost.
|
||
|
*/
|
||
|
private function discardBuffer() {
|
||
|
$str = $this->bufferOutputStr;
|
||
|
|
||
|
$this->bufferOutputStr = '';
|
||
|
$this->bufferOutput = false;
|
||
|
|
||
|
return $str;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Flushes the internal buffer,
|
||
|
* outputting what is left.
|
||
|
*
|
||
|
* @param append Optional, extra content to append onto the output buffer.
|
||
|
*/
|
||
|
private function flushBuffer() {
|
||
|
$temp = $this->bufferOutputStr;
|
||
|
$this->bufferOutputStr = '';
|
||
|
|
||
|
return $temp;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* This will finish buffering, and output the page.
|
||
|
* It also appends the magic JS onto the beginning of the page,
|
||
|
* if enabled, to allow working with Ajax.
|
||
|
*
|
||
|
* Note that if PHP Error has been disabled in the php.ini file,
|
||
|
* or through some other option, such as running from the command line,
|
||
|
* then this will do nothing (as no buffering will take place).
|
||
|
*/
|
||
|
public function endBuffer() {
|
||
|
if ( $this->isBufferSetup ) {
|
||
|
$content = ob_get_contents();
|
||
|
$handlers = ob_list_handlers();
|
||
|
|
||
|
$wasGZHandler = false;
|
||
|
|
||
|
$this->bufferOutput = true;
|
||
|
for ( $i = count($handlers)-1; $i >= 0; $i-- ) {
|
||
|
$handler = $handlers[$i];
|
||
|
|
||
|
if ( $handler === 'ob_gzhandler' ) {
|
||
|
$wasGZHandler = true;
|
||
|
ob_end_clean();
|
||
|
} else if ( $handler === 'default output handler' ) {
|
||
|
ob_end_clean();
|
||
|
} else {
|
||
|
ob_end_flush();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$content = $this->discardBuffer();
|
||
|
|
||
|
if ( $wasGZHandler ) {
|
||
|
ob_start('ob_gzhandler');
|
||
|
} else {
|
||
|
ob_start();
|
||
|
}
|
||
|
|
||
|
if (
|
||
|
!$this->isAjax &&
|
||
|
$this->catchAjaxErrors &&
|
||
|
(!$this->htmlOnly || !ErrorHandler::isNonPHPRequest()) &&
|
||
|
!ErrorHandler::isBinaryRequest()
|
||
|
) {
|
||
|
$js = $this->getContent( 'displayJSInjection' );
|
||
|
$js = JSMin::minify( $js );
|
||
|
|
||
|
// attemp to inject the script into the HTML, after the doctype
|
||
|
$matches = array();
|
||
|
preg_match( ErrorHandler::REGEX_DOCTYPE, $content, $matches );
|
||
|
|
||
|
if ( $matches ) {
|
||
|
$doctype = $matches[0];
|
||
|
$content = preg_replace( ErrorHandler::REGEX_DOCTYPE, "$doctype $js", $content );
|
||
|
} else {
|
||
|
echo $js;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
echo $content;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Calls the given method on this object,
|
||
|
* captures it's output, and then returns it.
|
||
|
*
|
||
|
* @param method The name of the method to call.
|
||
|
* @return All of the text outputted during the method call.
|
||
|
*/
|
||
|
private function getContent( $method ) {
|
||
|
ob_start();
|
||
|
$this->$method();
|
||
|
$content = ob_get_contents();
|
||
|
ob_end_clean();
|
||
|
|
||
|
return $content;
|
||
|
}
|
||
|
|
||
|
private function isApplicationFolder( $file ) {
|
||
|
return ErrorHandler::isFolderType(
|
||
|
$this->applicationFolders,
|
||
|
$this->applicationFoldersLongest,
|
||
|
$file
|
||
|
);
|
||
|
}
|
||
|
|
||
|
private function isIgnoreFolder( $file ) {
|
||
|
return ErrorHandler::isFolderType(
|
||
|
$this->ignoreFolders,
|
||
|
$this->ignoreFoldersLongest,
|
||
|
$file
|
||
|
);
|
||
|
}
|
||
|
|
||
|
private function getFolderType( $root, $file ) {
|
||
|
$testFile = $this->removeRootPath( $root, $file );
|
||
|
|
||
|
// it's this file : (
|
||
|
if ( $file === __FILE__ ) {
|
||
|
$type = ErrorHandler::FILE_TYPE_IGNORE;
|
||
|
} else if ( strpos($testFile, '/') === false ) {
|
||
|
$type = ErrorHandler::FILE_TYPE_ROOT;
|
||
|
} else if ( $this->isApplicationFolder($testFile) ) {
|
||
|
$type = ErrorHandler::FILE_TYPE_APPLICATION;
|
||
|
} else if ( $this->isIgnoreFolder($testFile) ) {
|
||
|
$type = ErrorHandler::FILE_TYPE_IGNORE;
|
||
|
} else {
|
||
|
$type = false;
|
||
|
}
|
||
|
|
||
|
return array( $type, $testFile );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Finds the file named, and returns it's contents in an array.
|
||
|
*
|
||
|
* It's essentially the same as 'file_get_contents'. However
|
||
|
* this will add caching at this PHP layer, avoiding lots of
|
||
|
* duplicate calls.
|
||
|
*
|
||
|
* It also splits the file into an array of lines, and makes
|
||
|
* it html safe.
|
||
|
*
|
||
|
* @param path The file to get the contents of.
|
||
|
* @return The file we are after, as an array of lines.
|
||
|
*/
|
||
|
private function getFileContents( $path ) {
|
||
|
if ( isset($this->cachedFiles[$path]) ) {
|
||
|
return $this->cachedFiles[$path];
|
||
|
} else {
|
||
|
$contents = @file_get_contents( $path );
|
||
|
|
||
|
if ( $contents ) {
|
||
|
$contents = explode(
|
||
|
"\n",
|
||
|
preg_replace(
|
||
|
'/(\r\n)|(\n\r)|\r/',
|
||
|
"\n",
|
||
|
str_replace( "\t", ' ', $contents )
|
||
|
)
|
||
|
);
|
||
|
|
||
|
$this->cachedFiles[ $path ] = $contents;
|
||
|
|
||
|
return $contents;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return array();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Reads out the code from the section of the line,
|
||
|
* which is at fault.
|
||
|
*
|
||
|
* The array is in a mapping of: array( line-number => line )
|
||
|
*
|
||
|
* If something goes wrong, then null is returned.
|
||
|
*/
|
||
|
private function readCodeFile( $errFile, $errLine ) {
|
||
|
try {
|
||
|
$lines = $this->getFileContents( $errFile );
|
||
|
|
||
|
if ( $lines ) {
|
||
|
$numLines = $this->numLines;
|
||
|
|
||
|
$searchUp = ceil( $numLines*0.75 );
|
||
|
$searchDown = $numLines - $searchUp;
|
||
|
|
||
|
$countLines = count( $lines );
|
||
|
|
||
|
/*
|
||
|
* Search around the errLine.
|
||
|
* We should aim get half of the lines above, and half from below.
|
||
|
* If that fails we get as many as we can.
|
||
|
*/
|
||
|
|
||
|
/*
|
||
|
* If we are near the bottom edge,
|
||
|
* we go down as far as we can,
|
||
|
* then work up the search area.
|
||
|
*/
|
||
|
if ( $errLine+$searchDown > $countLines ) {
|
||
|
$minLine = max( 0, $countLines-$numLines );
|
||
|
$maxLine = $countLines;
|
||
|
/*
|
||
|
* Go up as far as we can, up to half the search area.
|
||
|
* Then stretch down the whole search area.
|
||
|
*/
|
||
|
} else {
|
||
|
$minLine = max( 0, $errLine-$searchUp );
|
||
|
$maxLine = min( $minLine+$numLines, count($lines) );
|
||
|
}
|
||
|
|
||
|
$fileLines = array_splice( $lines, $minLine, $maxLine-$minLine );
|
||
|
|
||
|
$stripSize = -1;
|
||
|
foreach ( $fileLines as $i => $line ) {
|
||
|
$newLine = ltrim( $line, ' ' );
|
||
|
|
||
|
if ( strlen($newLine) > 0 ) {
|
||
|
$numSpaces = strlen($line) - strlen($newLine);
|
||
|
|
||
|
if ( $stripSize === -1 ) {
|
||
|
$stripSize = $numSpaces;
|
||
|
} else {
|
||
|
$stripSize = min( $stripSize, $numSpaces );
|
||
|
}
|
||
|
} else {
|
||
|
$fileLines[$i] = $newLine;
|
||
|
}
|
||
|
}
|
||
|
if ( $stripSize > 0 ) {
|
||
|
/*
|
||
|
* It's pretty common that PHP code is not flush with the left hand edge,
|
||
|
* so subtract 4 spaces, if we can,
|
||
|
* to account for this.
|
||
|
*/
|
||
|
if ( $stripSize > 4 ) {
|
||
|
$stripSize -= 4;
|
||
|
}
|
||
|
|
||
|
foreach ( $fileLines as $i => $line ) {
|
||
|
if ( strlen($line) > $stripSize ) {
|
||
|
$fileLines[$i] = substr( $line, $stripSize );
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$fileLines = join( "\n", $fileLines );
|
||
|
$fileLines = ErrorHandler::syntaxHighlight( $fileLines );
|
||
|
$fileLines = explode( "\n", $fileLines );
|
||
|
|
||
|
$lines = array();
|
||
|
for ( $i = 0; $i < count($fileLines); $i++ ) {
|
||
|
// +1 is because line numbers start at 1, whilst arrays start at 0
|
||
|
$lines[ $i+$minLine+1 ] = $fileLines[$i];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $lines;
|
||
|
} catch ( Exception $ex ) {
|
||
|
return null;
|
||
|
}
|
||
|
|
||
|
return null;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Attempts to remove the root path from the path given.
|
||
|
* If the path can't be removed, then the original path is returned.
|
||
|
*
|
||
|
* For example if root is 'C:/users/projects/my_site',
|
||
|
* and the file is 'C:/users/projects/my_site/index.php',
|
||
|
* then the root is removed, and we are left with just 'index.php'.
|
||
|
*
|
||
|
* This is to remove line noise; you don't need to be told the
|
||
|
* 'C:/whatever' bit 20 times.
|
||
|
*
|
||
|
* @param root The root path to remove.
|
||
|
* @param path The file we are removing the root section from.
|
||
|
*/
|
||
|
private function removeRootPath( $root, $path ) {
|
||
|
$filePath = str_replace( '\\', '/', $path );
|
||
|
|
||
|
if (
|
||
|
strpos($filePath, $root) === 0 &&
|
||
|
strlen($root) < strlen($filePath)
|
||
|
) {
|
||
|
return substr($filePath, strlen($root)+1 );
|
||
|
} else {
|
||
|
return $filePath;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Parses, and alters, the errLine, errFile and message given.
|
||
|
*
|
||
|
* This includes adding syntax highlighting, removing duplicate
|
||
|
* information we already have, and making the error easier to
|
||
|
* read.
|
||
|
*/
|
||
|
private function improveErrorMessage( $ex, $code, $message, $errLine, $errFile, $root, &$stackTrace ) {
|
||
|
// change these to change where the source file is come from
|
||
|
$srcErrFile = $errFile;
|
||
|
$srcErrLine = $errLine;
|
||
|
$altInfo = null;
|
||
|
$stackSearchI = 0;
|
||
|
|
||
|
$skipStackFirst = function( &$stackTrace ) {
|
||
|
$skipFirst = true;
|
||
|
|
||
|
foreach ( $stackTrace as $i => $trace ) {
|
||
|
if ( $skipFirst ) {
|
||
|
$skipFirst = false;
|
||
|
} else {
|
||
|
if ( $trace && isset($trace['file']) && isset($trace['line']) ) {
|
||
|
return array( $trace['file'], $trace['line'], $i );
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return array( null, null, null );
|
||
|
};
|
||
|
|
||
|
/*
|
||
|
* This is for calling a function that doesn't exist.
|
||
|
*
|
||
|
* The message contains a long description of where this takes
|
||
|
* place, even though we are already told this through line and
|
||
|
* file info. So we cut it out.
|
||
|
*/
|
||
|
if ( $code === 1 ) {
|
||
|
if (
|
||
|
( strpos($message, " undefined method ") !== false ) ||
|
||
|
( strpos($message, " undefined function ") !== false )
|
||
|
) {
|
||
|
$matches = array();
|
||
|
preg_match( '/\b[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*((->|::)[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*)?\\(\\)$/', $message, $matches );
|
||
|
|
||
|
/*
|
||
|
* undefined function or method call
|
||
|
*/
|
||
|
if ( $matches ) {
|
||
|
list( $className, $type, $functionName ) = ErrorHandler::splitFunction( $matches[0] );
|
||
|
|
||
|
if ( $stackTrace && isset($stackTrace[1]) && $stackTrace[1]['args'] ) {
|
||
|
$numArgs = count( $stackTrace[1]['args'] );
|
||
|
|
||
|
for ( $i = 0; $i < $numArgs; $i++ ) {
|
||
|
$args[]= ErrorHandler::newArgument( "_" );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$message = preg_replace(
|
||
|
'/\b[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*((->|::)[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*)?\\(\\)$/',
|
||
|
ErrorHandler::syntaxHighlightFunction( $className, $type, $functionName, $args ),
|
||
|
$message
|
||
|
);
|
||
|
}
|
||
|
} else if ( $message === 'Using $this when not in object context' ) {
|
||
|
$message = 'Using <span class="syntax-variable">$this</span> outside object context';
|
||
|
/*
|
||
|
* Class not found error.
|
||
|
*/
|
||
|
} else if (
|
||
|
strpos($message, "Class ") !== false &&
|
||
|
strpos($message, "not found") !== false
|
||
|
) {
|
||
|
$matches = array();
|
||
|
preg_match( '/\'(\\\\)?[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*((\\\\)?[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*)+\'/', $message, $matches );
|
||
|
|
||
|
if ( count($matches) > 0 ) {
|
||
|
// lose the 'quotes'
|
||
|
$className = $matches[0];
|
||
|
$className = substr( $className, 1, strlen($className)-2 );
|
||
|
|
||
|
$message = preg_replace(
|
||
|
'/\'(\\\\)?[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*((\\\\)?[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*)+\'/',
|
||
|
"<span class='syntax-class'>$className</span>",
|
||
|
$message
|
||
|
);
|
||
|
}
|
||
|
}
|
||
|
} else if ( $code === 2 ) {
|
||
|
if ( strpos($message, "Missing argument ") === 0 ) {
|
||
|
$message = preg_replace( '/, called in .*$/', '', $message );
|
||
|
|
||
|
$matches = array();
|
||
|
preg_match( ErrorHandler::REGEX_METHOD_OR_FUNCTION_END, $message, $matches );
|
||
|
|
||
|
if ( $matches ) {
|
||
|
$argumentMathces = array();
|
||
|
preg_match( '/^Missing argument ([0-9]+)/', $message, $argumentMathces );
|
||
|
$highlightArg = count($argumentMathces) === 2 ?
|
||
|
(((int) $argumentMathces[1])-1) :
|
||
|
null ;
|
||
|
|
||
|
$numHighlighted = 0;
|
||
|
$altInfo = ErrorHandler::syntaxHighlightFunctionMatch( $matches[0], $stackTrace, $highlightArg, $numHighlighted );
|
||
|
|
||
|
if ( $numHighlighted > 0 ) {
|
||
|
$message = preg_replace( '/^Missing argument ([0-9]+)/', 'Missing arguments ', $message );
|
||
|
}
|
||
|
|
||
|
if ( $altInfo ) {
|
||
|
$message = preg_replace( ErrorHandler::REGEX_METHOD_OR_FUNCTION_END, $altInfo, $message );
|
||
|
|
||
|
list( $srcErrFile, $srcErrLine, $stackSearchI ) = $skipStackFirst( $stackTrace );
|
||
|
}
|
||
|
}
|
||
|
} else if (
|
||
|
strpos($message, 'require(') === 0 ||
|
||
|
strpos($message, 'include(') === 0
|
||
|
) {
|
||
|
$endI = strpos( $message, '):' );
|
||
|
|
||
|
if ( $endI ) {
|
||
|
// include( is the same length
|
||
|
$requireLen = strlen('require(');
|
||
|
|
||
|
/*
|
||
|
* +2 to include the ): at the end of the string
|
||
|
*/
|
||
|
$postMessage = substr( $message, $endI+2 );
|
||
|
$postMessage = str_replace( 'failed to open stream: No ', 'no ', $postMessage );
|
||
|
$message = substr_replace( $message, $postMessage, $endI+2 );
|
||
|
|
||
|
/*
|
||
|
* If this string is in there, and where we think it should be,
|
||
|
* swap it with a shorter message.
|
||
|
*/
|
||
|
$replaceBit = 'failed to open stream: No ';
|
||
|
if ( strpos($message, $replaceBit) === $endI+2 ) {
|
||
|
$message = substr_replace( $message, 'no ', $endI+2, strlen($replaceBit) );
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Now put the string highlighting in there.
|
||
|
*/
|
||
|
$match = substr( $message, $requireLen, $endI-$requireLen );
|
||
|
$newString = "<span class='syntax-string'>'$match'</span>),";
|
||
|
$message = substr_replace( $message, $newString, $requireLen, ($endI-$requireLen)+2 );
|
||
|
}
|
||
|
}
|
||
|
/*
|
||
|
* Unexpected symbol errors.
|
||
|
* For example 'unexpected T_OBJECT_OPERATOR'.
|
||
|
*
|
||
|
* This swaps the 'T_WHATEVER' for the symbolic representation.
|
||
|
*/
|
||
|
} else if ( $code === 4 ) {
|
||
|
if ( $message === "syntax error, unexpected T_ENCAPSED_AND_WHITESPACE" ) {
|
||
|
$message = "syntax error, string is not closed";
|
||
|
} else {
|
||
|
$semiColonError = false;
|
||
|
if ( strpos($message, 'syntax error,') === 0 && $errLine > 2 ) {
|
||
|
$lines = ErrorHandler::getFileContents( $errFile );
|
||
|
|
||
|
$line = $lines[$errLine-1];
|
||
|
if ( preg_match( ErrorHandler::REGEX_MISSING_SEMI_COLON_FOLLOWING_LINE, $line ) !== 0 ) {
|
||
|
$content = rtrim( join( "\n", array_slice($lines, 0, $errLine-1) ) );
|
||
|
|
||
|
if ( strrpos($content, ';') !== strlen($content)-1 ) {
|
||
|
$message = "Missing semi-colon";
|
||
|
$errLine--;
|
||
|
$srcErrLine = $errLine;
|
||
|
$semiColonError = true;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( $semiColonError ) {
|
||
|
$matches = array();
|
||
|
$num = preg_match( '/\bunexpected ([A-Z_]+|\\$end)\b/', $message, $matches );
|
||
|
|
||
|
if ( $num > 0 ) {
|
||
|
$match = $matches[0];
|
||
|
$newSymbol = ErrorHandler::phpSymbolToDescription( str_replace('unexpected ', '', $match) );
|
||
|
|
||
|
$message = str_replace( $match, "unexpected $newSymbol", $message );
|
||
|
}
|
||
|
|
||
|
$matches = array();
|
||
|
$num = preg_match( '/, expecting ([A-Z_]+|\\$end)( or ([A-Z_]+|\\$end))*/', $message, $matches );
|
||
|
|
||
|
if ( $num > 0 ) {
|
||
|
$match = $matches[0];
|
||
|
$newMatch = str_replace( ", expecting ", '', $match );
|
||
|
$symbols = explode( ' or ', $newMatch );
|
||
|
foreach ( $symbols as $i => $sym ) {
|
||
|
$symbols[$i] = ErrorHandler::phpSymbolToDescription( $sym );
|
||
|
}
|
||
|
$newMatch = join( ', or ', $symbols );
|
||
|
|
||
|
$message = str_replace( $match, ", expecting $newMatch", $message );
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
/**
|
||
|
* Undefined Variable, add syntax highlighting and make variable from 'foo' too '$foo'.
|
||
|
*/
|
||
|
} else if ( $code === 8 ) {
|
||
|
if (
|
||
|
strpos($message, "Undefined variable:") !== false
|
||
|
) {
|
||
|
$matches = array();
|
||
|
preg_match( ErrorHandler::REGEX_VARIABLE, $message, $matches );
|
||
|
|
||
|
if ( count($matches) > 0 ) {
|
||
|
$message = 'Undefined variable <span class="syntax-variable">$' . $matches[0] . '</span>' ;
|
||
|
}
|
||
|
}
|
||
|
/**
|
||
|
* Invalid type given.
|
||
|
*/
|
||
|
} else if ( $code === 4096 ) {
|
||
|
if ( strpos($message, 'must be an ') ) {
|
||
|
$message = preg_replace( '/, called in .*$/', '', $message );
|
||
|
|
||
|
$matches = array();
|
||
|
preg_match( ErrorHandler::REGEX_METHOD_OR_FUNCTION, $message, $matches );
|
||
|
|
||
|
if ( $matches ) {
|
||
|
$argumentMathces = array();
|
||
|
preg_match( '/^Argument ([0-9]+)/', $message, $argumentMathces );
|
||
|
$highlightArg = count($argumentMathces) === 2 ?
|
||
|
(((int) $argumentMathces[1])-1) :
|
||
|
null ;
|
||
|
|
||
|
$fun = ErrorHandler::syntaxHighlightFunctionMatch( $matches[0], $stackTrace, $highlightArg );
|
||
|
|
||
|
if ( $fun ) {
|
||
|
$message = str_replace( 'passed to ', 'calling ', $message );
|
||
|
$message = preg_replace( ErrorHandler::REGEX_METHOD_OR_FUNCTION, $fun, $message );
|
||
|
$prioritizeCaller = true;
|
||
|
|
||
|
/*
|
||
|
* scalars not supported.
|
||
|
*/
|
||
|
$scalarType = null;
|
||
|
if ( ! ErrorHandler::$IS_SCALAR_TYPE_HINTING_SUPPORTED ) {
|
||
|
foreach ( ErrorHandler::$SCALAR_TYPES as $scalar ) {
|
||
|
if ( stripos($message, "must be an instance of $scalar,") !== false ) {
|
||
|
$scalarType = $scalar;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( $scalarType !== null ) {
|
||
|
$message = preg_replace( '/^Argument [0-9]+ calling /', 'Incorrect type hinting for ', $message );
|
||
|
$message = preg_replace(
|
||
|
'/ must be an instance of ' . ErrorHandler::REGEX_PHP_IDENTIFIER . '\b.*$/',
|
||
|
", ${scalarType} is not supported",
|
||
|
$message
|
||
|
);
|
||
|
|
||
|
$prioritizeCaller = false;
|
||
|
} else {
|
||
|
$message = preg_replace( '/ must be an (instance of )?' . ErrorHandler::REGEX_PHP_IDENTIFIER . '\b/', '', $message );
|
||
|
|
||
|
if ( preg_match('/, none given$/', $message) ) {
|
||
|
$message = preg_replace( '/^Argument /', 'Missing argument ', $message );
|
||
|
$message = preg_replace( '/, none given$/', '', $message );
|
||
|
} else {
|
||
|
$message = preg_replace( '/^Argument /', 'Incorrect argument ', $message );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( $prioritizeCaller ) {
|
||
|
list( $srcErrFile, $srcErrLine, $stackSearchI ) = $skipStackFirst( $stackTrace );
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( $stackTrace !== null ) {
|
||
|
$isEmpty = count( $stackTrace ) === 0 ;
|
||
|
|
||
|
if ( $isEmpty ) {
|
||
|
array_unshift( $stackTrace, array(
|
||
|
'line' => $errLine,
|
||
|
'file' => $errFile
|
||
|
) );
|
||
|
} else if (
|
||
|
count($stackTrace) > 0 && (
|
||
|
(! isset($stackTrace[0]['line'])) ||
|
||
|
($stackTrace[0]['line'] !== $errLine)
|
||
|
)
|
||
|
) {
|
||
|
array_unshift( $stackTrace, array(
|
||
|
'line' => $errLine,
|
||
|
'file' => $errFile
|
||
|
) );
|
||
|
}
|
||
|
|
||
|
if ( $stackTrace && !$isEmpty ) {
|
||
|
$ignoreCommons = false;
|
||
|
$len = count($stackTrace);
|
||
|
|
||
|
/*
|
||
|
* The code above can prioritize a location in the stack trace,
|
||
|
* this is 'stackSearchI'. So we should start our search from there,
|
||
|
* and work down the stack.
|
||
|
*
|
||
|
* This is built in a way so that when it reaches the end, it'll loop
|
||
|
* back round to the beginning, and check the traces we didn't check
|
||
|
* last time.
|
||
|
*
|
||
|
* If stackSearchI was not altered, then it just searches from top
|
||
|
* through to the bottom.
|
||
|
*/
|
||
|
for ( $i = $stackSearchI; $i < $stackSearchI+$len; $i++ ) {
|
||
|
$trace = &$stackTrace[ $i % $len ];
|
||
|
|
||
|
if ( isset($trace['file']) && isset($trace['line']) ) {
|
||
|
list( $type, $_ ) = $this->getFolderType( $root, $trace['file'] );
|
||
|
|
||
|
if ( $type !== ErrorHandler::FILE_TYPE_IGNORE ) {
|
||
|
if ( $type === ErrorHandler::FILE_TYPE_APPLICATION ) {
|
||
|
$srcErrLine = $trace['line'];
|
||
|
$srcErrFile = $trace['file'];
|
||
|
|
||
|
break;
|
||
|
} else if ( ! $ignoreCommons ) {
|
||
|
$srcErrLine = $trace['line'];
|
||
|
$srcErrFile = $trace['file'];
|
||
|
|
||
|
$ignoreCommons = true;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return array( $message, $srcErrFile, $srcErrLine, $altInfo );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Parses the stack trace, and makes it look pretty.
|
||
|
*
|
||
|
* This includes adding in the syntax highlighting,
|
||
|
* highlighting the colours for the files,
|
||
|
* and padding with whitespace.
|
||
|
*
|
||
|
* If stackTrace is null, then null is returned.
|
||
|
*/
|
||
|
private function parseStackTrace( $code, $message, $errLine, $errFile, &$stackTrace, $root, $altInfo=null ) {
|
||
|
if ( $stackTrace !== null ) {
|
||
|
/*
|
||
|
* For whitespace padding.
|
||
|
*/
|
||
|
$lineLen = 0;
|
||
|
$fileLen = 0;
|
||
|
|
||
|
// parse the stack trace, and remove the long urls
|
||
|
foreach ( $stackTrace as $i => $trace ) {
|
||
|
if ( $trace ) {
|
||
|
if ( isset($trace['line'] ) ) {
|
||
|
$lineLen = max( $lineLen, strlen($trace['line']) );
|
||
|
} else {
|
||
|
$trace['line'] = '';
|
||
|
}
|
||
|
|
||
|
$info = '';
|
||
|
|
||
|
if ( $i === 0 && $altInfo !== null ) {
|
||
|
$info = $altInfo;
|
||
|
/*
|
||
|
* Skip for the first iteration,
|
||
|
* as it's usually magical PHP calls.
|
||
|
*/
|
||
|
} else if (
|
||
|
$i > 0 && (
|
||
|
isset($trace['class']) ||
|
||
|
isset($trace['type']) ||
|
||
|
isset($trace['function'])
|
||
|
)
|
||
|
) {
|
||
|
$args = array();
|
||
|
if ( isset($trace['args']) ) {
|
||
|
foreach ( $trace['args'] as $arg ) {
|
||
|
$args[]= ErrorHandler::identifyTypeHTML( $arg, 1 );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$info = ErrorHandler::syntaxHighlightFunction(
|
||
|
isset($trace['class']) ? $trace['class'] : null,
|
||
|
isset($trace['type']) ? $trace['type'] : null,
|
||
|
isset($trace['function']) ? $trace['function'] : null,
|
||
|
$args
|
||
|
);
|
||
|
} else if ( isset($trace['info']) && $trace['info'] !== '' ) {
|
||
|
$info = ErrorHandler::syntaxHighlight( $trace['info'] );
|
||
|
} else if ( isset($trace['file']) && !isset($trace['info']) ) {
|
||
|
$contents = $this->getFileContents( $trace['file'] );
|
||
|
|
||
|
if ( $contents ) {
|
||
|
$info = ErrorHandler::syntaxHighlight(
|
||
|
trim( $contents[$trace['line']-1] )
|
||
|
);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$trace['info'] = $info;
|
||
|
|
||
|
if ( isset($trace['file']) ) {
|
||
|
list( $type, $file ) = $this->getFolderType( $root, $trace['file'] );
|
||
|
|
||
|
$trace['file_type'] = $type;
|
||
|
$trace['is_native'] = false;
|
||
|
} else {
|
||
|
$file = '[Internal PHP]';
|
||
|
|
||
|
$trace['file_type'] = '';
|
||
|
$trace['is_native'] = true;
|
||
|
}
|
||
|
|
||
|
$trace['file'] = $file;
|
||
|
|
||
|
$fileLen = max( $fileLen, strlen($file) );
|
||
|
|
||
|
$stackTrace[$i] = $trace;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* We are allowed to highlight just once, that's it.
|
||
|
*/
|
||
|
$highlightI = -1;
|
||
|
foreach ( $stackTrace as $i => $trace ) {
|
||
|
if (
|
||
|
$trace['line'] === $errLine &&
|
||
|
$trace['file'] === $errFile
|
||
|
) {
|
||
|
$highlightI = $i;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
foreach ( $stackTrace as $i => $trace ) {
|
||
|
if ( $trace ) {
|
||
|
// line
|
||
|
$line = str_pad( $trace['line'] , $lineLen, ' ', STR_PAD_LEFT );
|
||
|
|
||
|
// file
|
||
|
$file = $trace['file'];
|
||
|
$fileKlass = '';
|
||
|
if ( $trace['is_native'] ) {
|
||
|
$fileKlass = 'file-internal-php';
|
||
|
} else {
|
||
|
$fileKlass = 'filename ' . ErrorHandler::folderTypeToCSS( $trace['file_type'] );
|
||
|
}
|
||
|
$file = $file . str_pad( '', $fileLen-strlen($file), ' ', STR_PAD_LEFT );
|
||
|
|
||
|
// info
|
||
|
$info = $trace['info'];
|
||
|
if ( $info ) {
|
||
|
$info = str_replace( "\n", '\n', $info );
|
||
|
$info = str_replace( "\r", '\r', $info );
|
||
|
} else {
|
||
|
$info = ' ';
|
||
|
}
|
||
|
|
||
|
// line + file + info
|
||
|
$file = trim( $file );
|
||
|
|
||
|
$stackStr =
|
||
|
"<td class='linenumber'>$line</td>" .
|
||
|
"<td class='$fileKlass'>$file</td>" .
|
||
|
"<td class='lineinfo'>$info</td>" ;
|
||
|
|
||
|
if ( $trace['is_native'] ) {
|
||
|
$cssClass = 'is-native ';
|
||
|
} else {
|
||
|
$cssClass = '';
|
||
|
}
|
||
|
|
||
|
if ( $highlightI === $i ) {
|
||
|
$cssClass .= ' highlight';
|
||
|
} else if ( $highlightI > $i ) {
|
||
|
$cssClass .= ' pre-highlight';
|
||
|
}
|
||
|
|
||
|
if (
|
||
|
$i !== 0 &&
|
||
|
isset($trace['exception']) &&
|
||
|
$trace['exception']
|
||
|
) {
|
||
|
$ex = $trace['exception'];
|
||
|
|
||
|
$exHtml = '<tr class="error-stack-trace-exception"><td>' .
|
||
|
'exception "' .
|
||
|
htmlspecialchars( $ex->getMessage() ) .
|
||
|
'"' .
|
||
|
'</td></tr>';
|
||
|
} else {
|
||
|
$exHtml = '';
|
||
|
}
|
||
|
|
||
|
$data = '';
|
||
|
if ( isset($trace['file-id']) ) {
|
||
|
$data = ' data-file-id="' . $trace['file-id'] . '"' .
|
||
|
' data-line="' . $line . '"' ;
|
||
|
}
|
||
|
|
||
|
$stackTrace[$i] = "$exHtml<tr class='error-stack-trace-line $cssClass' $data>$stackStr</tr>";
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return '<table id="error-stack-trace">' . join( "", $stackTrace ) . '</table>';
|
||
|
} else {
|
||
|
return null;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private function logError( $message, $file, $line, $ex=null ) {
|
||
|
if ( $ex ) {
|
||
|
$trace = $ex->getTraceAsString();
|
||
|
$parts = explode( "\n", $trace );
|
||
|
$trace = " " . join( "\n ", $parts );
|
||
|
|
||
|
if ( ! ErrorHandler::isIIS() ) {
|
||
|
error_log( "$message \n $file, $line \n$trace" );
|
||
|
}
|
||
|
} else {
|
||
|
if ( ! ErrorHandler::isIIS() ) {
|
||
|
error_log( "$message \n $file, $line" );
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Given a class name, which can include a namespace,
|
||
|
* this will report that it is not found.
|
||
|
*
|
||
|
* This will also report it as an exception,
|
||
|
* so you will get a full stack trace.
|
||
|
*/
|
||
|
public function reportClassNotFound( $className ) {
|
||
|
throw new \ErrorException( "Class '$className' not found", E_ERROR, 0, __FILE__, __LINE__ );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Given an exception, this will report it.
|
||
|
*/
|
||
|
public function reportException( $ex ) {
|
||
|
$this->reportError(
|
||
|
$ex->getCode(),
|
||
|
$ex->getMessage(),
|
||
|
$ex->getLine(),
|
||
|
$ex->getFile(),
|
||
|
$ex
|
||
|
);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* The entry point for handling an error.
|
||
|
*
|
||
|
* This is the lowest entry point for error reporting,
|
||
|
* and for that reason it can either take just error info,
|
||
|
* or a combination of error and exception information.
|
||
|
*
|
||
|
* Note that this will still log errors in the error log
|
||
|
* even when it's disabled with ini. It just does nothing
|
||
|
* more than that.
|
||
|
*/
|
||
|
public function reportError( $code, $message, $errLine, $errFile, $ex=null ) {
|
||
|
$this->discardBuffer();
|
||
|
|
||
|
if (
|
||
|
$ex === null &&
|
||
|
$code === 1 &&
|
||
|
strpos($message, "Class ") === 0 &&
|
||
|
strpos($message, "not found") !== false &&
|
||
|
$this->classNotFoundException !== null
|
||
|
) {
|
||
|
$ex = $this->classNotFoundException;
|
||
|
|
||
|
$code = $ex->getCode();
|
||
|
$message = $ex->getMessage();
|
||
|
$errLine = $ex->getLine();
|
||
|
$errFile = $ex->getFile();
|
||
|
$stackTrace = $ex->getTrace();
|
||
|
}
|
||
|
|
||
|
$this->logError( $message, $errFile, $errLine, $ex );
|
||
|
|
||
|
/**
|
||
|
* It runs if:
|
||
|
* - it is globally enabled
|
||
|
* - this error handler is enabled
|
||
|
* - we believe it is a regular html request, or ajax
|
||
|
*/
|
||
|
global $_php_error_is_ini_enabled;
|
||
|
if (
|
||
|
$_php_error_is_ini_enabled &&
|
||
|
$this->isOn() && (
|
||
|
$this->isAjax ||
|
||
|
!$this->htmlOnly ||
|
||
|
!ErrorHandler::isNonPHPRequest()
|
||
|
)
|
||
|
) {
|
||
|
$root = $this->applicationRoot;
|
||
|
|
||
|
list( $ex, $stackTrace, $code, $errFile, $errLine ) =
|
||
|
$this->getStackTrace( $ex, $code, $errFile, $errLine );
|
||
|
|
||
|
list( $message, $srcErrFile, $srcErrLine, $altInfo ) =
|
||
|
$this->improveErrorMessage(
|
||
|
$ex,
|
||
|
$code,
|
||
|
$message,
|
||
|
$errLine,
|
||
|
$errFile,
|
||
|
$root,
|
||
|
$stackTrace
|
||
|
);
|
||
|
|
||
|
$errFile = $srcErrFile;
|
||
|
$errLine = $srcErrLine;
|
||
|
|
||
|
list( $fileLinesSets, $numFileLines ) = $this->generateFileLineSets( $srcErrFile, $srcErrLine, $stackTrace );
|
||
|
|
||
|
list( $type, $errFile ) = $this->getFolderType( $root, $errFile );
|
||
|
$errFileType = ErrorHandler::folderTypeToCSS( $type );
|
||
|
|
||
|
$stackTrace = $this->parseStackTrace( $code, $message, $errLine, $errFile, $stackTrace, $root, $altInfo );
|
||
|
$fileLines = $this->readCodeFile( $srcErrFile, $srcErrLine );
|
||
|
|
||
|
// load the session, if ...
|
||
|
// - there *is* a session cookie to load
|
||
|
// - the session has not yet been started
|
||
|
// Do not start the session without he cookie, because there may be no session ever.
|
||
|
if ( isset($_COOKIE[session_name()]) && session_id() === '' ) {
|
||
|
session_start();
|
||
|
}
|
||
|
|
||
|
$request = ErrorHandler::getRequestHeaders();
|
||
|
$response = ErrorHandler::getResponseHeaders();
|
||
|
|
||
|
$dump = $this->generateDumpHTML(
|
||
|
array(
|
||
|
'post' => ( isset($_POST) ? $_POST : array() ),
|
||
|
'get' => ( isset($_GET) ? $_GET : array() ),
|
||
|
'session' => ( isset($_SESSION) ? $_SESSION : array() ),
|
||
|
'cookies' => ( isset($_COOKIE) ? $_COOKIE : array() )
|
||
|
),
|
||
|
|
||
|
$request,
|
||
|
$response,
|
||
|
|
||
|
$_SERVER
|
||
|
);
|
||
|
$this->displayError( $message, $srcErrLine, $errFile, $errFileType, $stackTrace, $fileLinesSets, $numFileLines, $dump );
|
||
|
|
||
|
// exit in order to end processing
|
||
|
$this->turnOff();
|
||
|
exit(0);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private function getStackTrace( $ex, $code, $errFile, $errLine ) {
|
||
|
$stackTrace = null;
|
||
|
|
||
|
if ( $ex !== null ) {
|
||
|
$next = $ex;
|
||
|
$stackTrace = array();
|
||
|
$skipStacks = 0;
|
||
|
|
||
|
for (
|
||
|
$next = $ex;
|
||
|
$next !== null;
|
||
|
$next = $next->getPrevious()
|
||
|
) {
|
||
|
$ex = $next;
|
||
|
|
||
|
$stack = $ex->getTrace();
|
||
|
$file = $ex->getFile();
|
||
|
$line = $ex->getLine();
|
||
|
|
||
|
if ( $stackTrace !== null && count($stackTrace) > 0 ) {
|
||
|
$stack = array_slice( $stack, 0, count($stack)-count($stackTrace) + 1 );
|
||
|
}
|
||
|
|
||
|
if ( count($stack) > 0 && (
|
||
|
!isset($stack[0]['file']) ||
|
||
|
!isset($stack[0]['line']) ||
|
||
|
$stack[0]['file'] !== $file ||
|
||
|
$stack[0]['line'] !== $line
|
||
|
) ) {
|
||
|
array_unshift( $stack, array(
|
||
|
'file' => $file,
|
||
|
'line' => $line
|
||
|
) );
|
||
|
}
|
||
|
|
||
|
$stackTrace = ( $stackTrace !== null ) ?
|
||
|
array_merge( $stack, $stackTrace ) :
|
||
|
$stack ;
|
||
|
|
||
|
if ( count($stackTrace) > 0 ) {
|
||
|
$stackTrace[0]['exception'] = $ex;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$message = $ex->getMessage();
|
||
|
$errFile = $ex->getFile();
|
||
|
$errLine = $ex->getLine();
|
||
|
|
||
|
$code = $ex->getCode();
|
||
|
|
||
|
if ( method_exists($ex, 'getSeverity') ) {
|
||
|
$severity = $ex->getSeverity();
|
||
|
|
||
|
if ( $code === 0 && $severity !== 0 && $severity !== null ) {
|
||
|
$code = $severity;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return array( $ex, $stackTrace, $code, $errFile, $errLine );
|
||
|
}
|
||
|
|
||
|
private function generateDumpHTML( $arrays, $request, $response, $server ) {
|
||
|
$arrToHtml = function( $name, $array, $css='' ) {
|
||
|
$max = 0;
|
||
|
|
||
|
foreach ( $array as $e => $v ) {
|
||
|
$max = max( $max, strlen( $e ) );
|
||
|
}
|
||
|
|
||
|
$snippet = "<h2 class='error_dump_header'>$name</h2>";
|
||
|
|
||
|
foreach ( $array as $e => $v ) {
|
||
|
$e = str_pad( $e, $max, ' ', STR_PAD_RIGHT );
|
||
|
|
||
|
$e = htmlentities( $e );
|
||
|
$v = ErrorHandler::identifyTypeHTML( $v, 3 );
|
||
|
|
||
|
$snippet .= "<div class='error_dump_key'>$e</div><div class='error_dump_mapping'>=></div><div class='error_dump_value'>$v</div>";
|
||
|
}
|
||
|
|
||
|
return "<div class='error_dump $css'>$snippet</div>";
|
||
|
};
|
||
|
|
||
|
$html = '';
|
||
|
foreach ( $arrays as $key => $value ) {
|
||
|
if ( isset($value) && $value ) {
|
||
|
$html .= $arrToHtml( $key, $value );
|
||
|
} else {
|
||
|
unset($arrays[$key]);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return "<div class='error-dumps'>" .
|
||
|
$html .
|
||
|
$arrToHtml( 'request', $request, 'dump_request' ) .
|
||
|
$arrToHtml( 'response', $response, 'dump_response' ) .
|
||
|
$arrToHtml( 'server', $server, 'dump_server' ) .
|
||
|
"</div>";
|
||
|
}
|
||
|
|
||
|
private function generateFileLineSets( $srcErrFile, $srcErrLine, &$stackTrace ) {
|
||
|
$fileLineID = 1;
|
||
|
$srcErrID = "file-line-$fileLineID";
|
||
|
$fileLineID++;
|
||
|
|
||
|
|
||
|
$lines = $this->getFileContents( $srcErrFile );
|
||
|
$minSize = count( $lines );
|
||
|
|
||
|
$srcFileSet = new FileLinesSet( $srcErrFile, $srcErrID, $lines );
|
||
|
|
||
|
$seenFiles = array( $srcErrFile => $srcFileSet );
|
||
|
|
||
|
if ( $stackTrace ) {
|
||
|
foreach ( $stackTrace as $i => &$trace ) {
|
||
|
if ( $trace && isset($trace['file']) && isset($trace['line']) ) {
|
||
|
$file = $trace['file'];
|
||
|
$line = $trace['line'];
|
||
|
|
||
|
if ( isset($seenFiles[$file]) ) {
|
||
|
$fileSet = $seenFiles[$file];
|
||
|
} else {
|
||
|
$traceFileID = "file-line-$fileLineID";
|
||
|
|
||
|
$lines = $this->getFileContents( $file );
|
||
|
$minSize = max( $minSize, count($lines) );
|
||
|
$fileSet = new FileLinesSet( $file, $traceFileID, $lines );
|
||
|
|
||
|
$seenFiles[ $file ] = $fileSet;
|
||
|
|
||
|
$fileLineID++;
|
||
|
}
|
||
|
|
||
|
$trace['file-id'] = $fileSet->getHTMLID();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return array( array_values($seenFiles), $minSize );
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Even if disabled, we still act like reporting is on,
|
||
|
* if it's turned on.
|
||
|
*
|
||
|
* We just don't do anything.
|
||
|
*/
|
||
|
private function setEnabled( $isOn ) {
|
||
|
$wasOn = $this->isOn;
|
||
|
$this->isOn = $isOn;
|
||
|
|
||
|
global $_php_error_is_ini_enabled;
|
||
|
if ( $_php_error_is_ini_enabled ) {
|
||
|
/*
|
||
|
* Only turn off, if we're moving from on to off.
|
||
|
*
|
||
|
* This is so if it's turned off without turning on,
|
||
|
* we don't change anything.
|
||
|
*/
|
||
|
if ( !$isOn ) {
|
||
|
if ( $wasOn ) {
|
||
|
$this->runDisableErrors();
|
||
|
}
|
||
|
/*
|
||
|
* Always turn it on, even if already on.
|
||
|
*
|
||
|
* This is incase it was messed up in some way
|
||
|
* by the user.
|
||
|
*/
|
||
|
} else if ( $isOn ) {
|
||
|
$this->runEnableErrors();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private function runDisableErrors() {
|
||
|
global $_php_error_is_ini_enabled;
|
||
|
|
||
|
if ( $_php_error_is_ini_enabled ) {
|
||
|
error_reporting( $this->defaultErrorReportingOff );
|
||
|
|
||
|
@ini_restore( 'html_errors' );
|
||
|
|
||
|
if ( ErrorHandler::isIIS() ) {
|
||
|
@ini_restore( 'log_errors' );
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Now the actual hooking into PHP's error reporting.
|
||
|
*
|
||
|
* We enable _ALL_ errors, and make them all exceptions.
|
||
|
* We also need to hook into the shutdown function so
|
||
|
* we can catch fatal and compile time errors.
|
||
|
*/
|
||
|
private function runEnableErrors() {
|
||
|
global $_php_error_is_ini_enabled;
|
||
|
|
||
|
if ( $_php_error_is_ini_enabled ) {
|
||
|
$catchSurpressedErrors = &$this->catchSurpressedErrors;
|
||
|
$self = $this;
|
||
|
|
||
|
// all errors \o/ !
|
||
|
error_reporting( $this->defaultErrorReportingOn );
|
||
|
@ini_set( 'html_errors', false );
|
||
|
|
||
|
if ( ErrorHandler::isIIS() ) {
|
||
|
@ini_set( 'log_errors', false );
|
||
|
}
|
||
|
|
||
|
set_error_handler(
|
||
|
function( $code, $message, $file, $line, $context ) use ( $self, &$catchSurpressedErrors ) {
|
||
|
/*
|
||
|
* DO NOT! log the error.
|
||
|
*
|
||
|
* Either it's thrown as an exception, and so logged by the exception handler,
|
||
|
* or we return false, and it's logged by PHP.
|
||
|
*
|
||
|
* Also DO NOT! throw an exception, instead report it.
|
||
|
* This is because if an operation raises both a user AND
|
||
|
* fatal error (such as require), then the exception is
|
||
|
* silently ignored.
|
||
|
*/
|
||
|
if ( $self->isOn() ) {
|
||
|
/*
|
||
|
* When using an @, the error reporting drops to 0.
|
||
|
*/
|
||
|
if ( error_reporting() !== 0 || $catchSurpressedErrors ) {
|
||
|
$ex = new \ErrorException( $message, $code, 0, $file, $line );
|
||
|
|
||
|
$self->reportException( $ex );
|
||
|
}
|
||
|
} else {
|
||
|
return false;
|
||
|
}
|
||
|
},
|
||
|
$this->defaultErrorReportingOn
|
||
|
);
|
||
|
|
||
|
set_exception_handler( function($ex) use ( $self ) {
|
||
|
if ( $self->isOn() ) {
|
||
|
$self->reportException( $ex );
|
||
|
} else {
|
||
|
return false;
|
||
|
}
|
||
|
});
|
||
|
|
||
|
if ( ! $self->isShutdownRegistered ) {
|
||
|
if ( $self->catchClassNotFound ) {
|
||
|
$classException = &$self->classNotFoundException;
|
||
|
$autoloaderFuns = ErrorHandler::$SAFE_AUTOLOADER_FUNCTIONS;
|
||
|
|
||
|
/*
|
||
|
* When this is called, the key point is that we don't error!
|
||
|
*
|
||
|
* Instead we record that an error has occurred,
|
||
|
* if we believe one has, and then let PHP error as normal.
|
||
|
* The stack trace we record is then used later.
|
||
|
*
|
||
|
* This is done for two reasons:
|
||
|
* - functions like 'class_exists' will run the autoloader, and we shouldn't error on them
|
||
|
* - on PHP 5.3.0, the class loader registered functions does *not* return closure objects, so we can't do anything clever.
|
||
|
*
|
||
|
* So we watch, but don't touch.
|
||
|
*/
|
||
|
spl_autoload_register( function($className) use ( $self, &$classException, &$autoloaderFuns ) {
|
||
|
if ( $self->isOn() ) {
|
||
|
$classException = null;
|
||
|
|
||
|
// search the stack first, to check if we are running from 'class_exists' before we error
|
||
|
if ( defined('DEBUG_BACKTRACE_IGNORE_ARGS') ) {
|
||
|
$trace = debug_backtrace( DEBUG_BACKTRACE_IGNORE_ARGS );
|
||
|
} else {
|
||
|
$trace = debug_backtrace();
|
||
|
}
|
||
|
$error = true;
|
||
|
|
||
|
foreach ( $trace as $row ) {
|
||
|
if ( isset($row['function']) ) {
|
||
|
$function = $row['function'];
|
||
|
|
||
|
// they are just checking, so don't error
|
||
|
if ( in_array($function, $autoloaderFuns, true) ) {
|
||
|
$error = false;
|
||
|
break;
|
||
|
// not us, and not the autoloader, so error!
|
||
|
} else if (
|
||
|
$function !== '__autoload' &&
|
||
|
$function !== 'spl_autoload_call' &&
|
||
|
strpos($function, 'php_error\\') === false
|
||
|
) {
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( $error ) {
|
||
|
$classException = new \ErrorException( "Class '$className' not found", E_ERROR, 0, __FILE__, __LINE__ );
|
||
|
}
|
||
|
}
|
||
|
} );
|
||
|
}
|
||
|
|
||
|
$self->isShutdownRegistered = true;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private function displayJSInjection() {
|
||
|
?><script data-php_error="magic JS, just ignore this!">
|
||
|
"use strict";
|
||
|
|
||
|
(function( window ) {
|
||
|
if ( window.XMLHttpRequest ) {
|
||
|
/**
|
||
|
* A method wrapping helper function.
|
||
|
*
|
||
|
* Wraps the method given, from the old prototype to the new
|
||
|
* XMLHttpRequest prototype.
|
||
|
*
|
||
|
* This only happens if the old one actually has that prototype.
|
||
|
* If the browser doesn't support a prototype, then it doesn't
|
||
|
* get wrapped.
|
||
|
*/
|
||
|
var wrapMethod = function( XMLHttpRequest, old, prop ) {
|
||
|
if ( old.prototype[prop] ) {
|
||
|
var behaviours = ( arguments.length > 3 ) ?
|
||
|
Array.prototype.slice.call( arguments, 3 ) :
|
||
|
null ;
|
||
|
|
||
|
XMLHttpRequest.prototype[prop] = function() {
|
||
|
if ( behaviours !== null ) {
|
||
|
for ( var i = 0; i < behaviours.length; i++ ) {
|
||
|
behaviours[i].call( this, arguments, prop );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return this.__.inner[prop].apply( this.__.inner, arguments );
|
||
|
};
|
||
|
}
|
||
|
}
|
||
|
|
||
|
var postMethod = function( XMLHttpRequest, prop ) {
|
||
|
if ( XMLHttpRequest.prototype[prop] ) {
|
||
|
var behaviours = Array.prototype.slice.call( arguments, 2 );
|
||
|
|
||
|
var previous = XMLHttpRequest.prototype[prop];
|
||
|
XMLHttpRequest.prototype[prop] = function() {
|
||
|
var r = previous.apply( this, arguments );
|
||
|
|
||
|
for ( var i = 0; i < behaviours.length; i++ ) {
|
||
|
behaviours[i].call( this, arguments, prop );
|
||
|
}
|
||
|
|
||
|
return r;
|
||
|
};
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Certain properties will error when read,
|
||
|
* and which ones do vary from browser to browser.
|
||
|
*
|
||
|
* I've found both Chrome and Firefox will error
|
||
|
* on _different_ properties.
|
||
|
*
|
||
|
* So every read needs to be wrapped in a try/catch,
|
||
|
* and just hope it doesn't error.
|
||
|
*/
|
||
|
var copyProperties = function( src, dest, props ) {
|
||
|
for ( var i = 0; i < props.length; i++ ) {
|
||
|
try {
|
||
|
var prop = props[i];
|
||
|
dest[prop] = src[prop];
|
||
|
} catch( ex ) { }
|
||
|
}
|
||
|
};
|
||
|
|
||
|
var copyResponseProperties = function( src, dest ) {
|
||
|
copyProperties( src, dest, [
|
||
|
'response',
|
||
|
'responseText',
|
||
|
'responseXML'
|
||
|
]);
|
||
|
}
|
||
|
|
||
|
var copyRequestProperties = function( src, dest, includeReadOnly, skipResponse ) {
|
||
|
copyProperties( src, dest, [
|
||
|
'readyState',
|
||
|
'timeout',
|
||
|
'upload',
|
||
|
'withCredentials',
|
||
|
'responseType',
|
||
|
|
||
|
'mozBackgroundRequest',
|
||
|
'mozArrayBuffer',
|
||
|
'multipart'
|
||
|
]);
|
||
|
|
||
|
if ( includeReadOnly ) {
|
||
|
copyProperties( src, dest, [
|
||
|
'status',
|
||
|
'statusText',
|
||
|
'channel'
|
||
|
]);
|
||
|
|
||
|
if ( ! skipResponse ) {
|
||
|
copyResponseProperties( src, dest );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return dest;
|
||
|
}
|
||
|
|
||
|
var runFail = function( ev ) {
|
||
|
var self = this;
|
||
|
var xmlHttpRequest = this.__.inner;
|
||
|
|
||
|
var iframe = document.createElement('iframe');
|
||
|
iframe.setAttribute('width', '100%');
|
||
|
iframe.setAttribute('height', '100%');
|
||
|
iframe.setAttribute('src', 'about:blank');
|
||
|
|
||
|
iframe.style.transition =
|
||
|
iframe.style.OTransition =
|
||
|
iframe.style.MsTransition =
|
||
|
iframe.style.MozTransition =
|
||
|
iframe.style.WebkitTransition = 'opacity 200ms linear';
|
||
|
|
||
|
iframe.style.background = 'transparent';
|
||
|
iframe.style.opacity = 0;
|
||
|
iframe.style.zIndex = 100001;
|
||
|
iframe.style.top = 0;
|
||
|
iframe.style.right = 0;
|
||
|
iframe.style.left = 0;
|
||
|
iframe.style.bottom = 0;
|
||
|
iframe.style.position = 'fixed';
|
||
|
|
||
|
var response = xmlHttpRequest.responseText;
|
||
|
|
||
|
iframe.onload = function() {
|
||
|
var iDoc = iframe.contentWindow || iframe.contentDocument;
|
||
|
if ( iDoc.document) {
|
||
|
iDoc = iDoc.document;
|
||
|
}
|
||
|
|
||
|
var iBody = iDoc.getElementsByTagName("body")[0];
|
||
|
iBody.innerHTML = response;
|
||
|
var iHead = iDoc.getElementsByTagName("head")[0];
|
||
|
|
||
|
// re-run the script tags
|
||
|
var scripts = iDoc.getElementsByTagName('script');
|
||
|
for ( var i = 0; i < scripts.length; i++ ) {
|
||
|
var script = scripts[i];
|
||
|
var parent = script.parentNode;
|
||
|
|
||
|
if ( parent ) {
|
||
|
parent.removeChild( script );
|
||
|
|
||
|
var newScript = iDoc.createElement('script');
|
||
|
newScript.innerHTML = script.innerHTML;
|
||
|
|
||
|
iHead.appendChild( newScript );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
var closed = false;
|
||
|
var closeIFrame = function() {
|
||
|
if ( ! closed ) {
|
||
|
closed = true;
|
||
|
|
||
|
iframe.style.opacity = 0;
|
||
|
|
||
|
setTimeout( function() {
|
||
|
iframe.parentNode.removeChild( iframe );
|
||
|
}, 220 );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Retry Handler.
|
||
|
*
|
||
|
* Clear this, make a new (real) XMLHttpRequest,
|
||
|
* and then re-run everything.
|
||
|
*/
|
||
|
var retry = iDoc.getElementById('ajax-retry');
|
||
|
if ( retry ) {
|
||
|
var retryFun = function() {
|
||
|
var methodCalls = self.__.methodCalls;
|
||
|
|
||
|
initializeXMLHttpRequest.call( self );
|
||
|
for ( var i = 0; i < methodCalls.length; i++ ) {
|
||
|
var method = methodCalls[i];
|
||
|
self[method.method].apply( self, method.args );
|
||
|
}
|
||
|
|
||
|
closeIFrame();
|
||
|
|
||
|
return false;
|
||
|
};
|
||
|
retry.onclick = retryFun;
|
||
|
|
||
|
iframe.__php_error_retry = retryFun;
|
||
|
|
||
|
/*
|
||
|
* The close handler.
|
||
|
*
|
||
|
* When closed, the response is cleared,
|
||
|
* and then the request finishes with null info.
|
||
|
*/
|
||
|
iDoc.getElementById('ajax-close').onclick = function() {
|
||
|
copyRequestProperties( self.__.inner, self, true );
|
||
|
|
||
|
// clear the response
|
||
|
self.response = '';
|
||
|
self.responseText = '';
|
||
|
self.responseXML = null;
|
||
|
|
||
|
if ( self.onreadystatechange ) {
|
||
|
self.onreadystatechange( ev );
|
||
|
}
|
||
|
|
||
|
closeIFrame();
|
||
|
return false;
|
||
|
};
|
||
|
|
||
|
var html = iDoc.getElementsByTagName('html')[0];
|
||
|
html.setAttribute( 'class', 'ajax' );
|
||
|
|
||
|
setTimeout( function() {
|
||
|
iframe.style.opacity = 1;
|
||
|
}, 1 );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Placed inside a timeout, incase the document doesn't exist yet.
|
||
|
*
|
||
|
* Can happen if the page ajax's straight away.
|
||
|
*/
|
||
|
setTimeout( function() {
|
||
|
var body = document.getElementsByTagName('body')[0];
|
||
|
body.appendChild( iframe );
|
||
|
}, 1 );
|
||
|
}
|
||
|
|
||
|
var old = window.XMLHttpRequest;
|
||
|
|
||
|
/**
|
||
|
* The middle man http request object.
|
||
|
*
|
||
|
* Acts just like a normal one, but will show errors if they
|
||
|
* occur instead of running the result.
|
||
|
*/
|
||
|
var XMLHttpRequest = function() {
|
||
|
initializeXMLHttpRequest.call( this );
|
||
|
}
|
||
|
|
||
|
var initializeXMLHttpRequest = function() {
|
||
|
var self = this,
|
||
|
inner = new old();
|
||
|
|
||
|
/**
|
||
|
* With a buggy XMLHttpRequest, it's possible to accidentally run the error handler
|
||
|
* multiple times.
|
||
|
*
|
||
|
* This is a flag to only do it once, to keep the code more defensive.
|
||
|
*/
|
||
|
var errorOnce = true,
|
||
|
isAjaxError = false;
|
||
|
|
||
|
var stateResults = [];
|
||
|
|
||
|
inner.onreadystatechange = function( ev ) {
|
||
|
copyRequestProperties( inner, self, true, true );
|
||
|
|
||
|
var state = inner.readyState;
|
||
|
|
||
|
/*
|
||
|
* Check headers for error.
|
||
|
*/
|
||
|
if ( ! isAjaxError && state >= 2 ) {
|
||
|
/*
|
||
|
* It's null in some browsers, and an empty string in others.
|
||
|
*/
|
||
|
var header = inner.getResponseHeader( '<?php echo ErrorHandler::PHP_ERROR_MAGIC_HEADER_KEY ?>' );
|
||
|
|
||
|
if ( header !== null && header !== '' ) {
|
||
|
self.__.isAjaxError = true;
|
||
|
isAjaxError = true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( ! isAjaxError && state >= 2 ) {
|
||
|
copyResponseProperties( inner, self );
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Success ! \o/
|
||
|
*
|
||
|
* Pass any state change on to the parent caller,
|
||
|
* unless we hit an ajaxy error.
|
||
|
*/
|
||
|
if ( !isAjaxError && self.onreadystatechange ) {
|
||
|
/*
|
||
|
* One of three things happens:
|
||
|
* = cache the requests until we know there is no error (state 4)
|
||
|
* = we know there is no error, and so we run our cache
|
||
|
* = cache is done, but we've been called again, so just pass it on
|
||
|
*/
|
||
|
if ( state < 4 ) {
|
||
|
stateResults.push( copyRequestProperties(self, {}, true) );
|
||
|
} else {
|
||
|
if ( stateResults !== null ) {
|
||
|
var currentState = copyRequestProperties( self, {}, true );
|
||
|
|
||
|
for ( var i = 0; i < stateResults.length; i++ ) {
|
||
|
var store = stateResults[i];
|
||
|
copyRequestProperties( store, self, true );
|
||
|
|
||
|
// must check a second time here,
|
||
|
// in case it gets changed within an onreadystatechange
|
||
|
if ( self.onreadystatechange ) {
|
||
|
self.onreadystatechange( ev );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
copyRequestProperties( currentState, self, true );
|
||
|
stateResults = null;
|
||
|
}
|
||
|
|
||
|
if ( self.onreadystatechange ) {
|
||
|
self.onreadystatechange( ev );
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Fail : (
|
||
|
*/
|
||
|
if (
|
||
|
isAjaxError &&
|
||
|
state === 4 &&
|
||
|
errorOnce
|
||
|
) {
|
||
|
errorOnce = false;
|
||
|
if ( window.console && window.console.log ) {
|
||
|
window.console.log( 'Ajax Error Calling: ' + self.__.url );
|
||
|
}
|
||
|
|
||
|
runFail.call( self, ev );
|
||
|
}
|
||
|
};
|
||
|
|
||
|
copyRequestProperties( inner, this, true );
|
||
|
|
||
|
/*
|
||
|
* Private fields are stored underneath the unhappy face,
|
||
|
* to localize them.
|
||
|
*
|
||
|
* Access becomes:
|
||
|
* this.__.fieldName
|
||
|
*/
|
||
|
this.__ = {
|
||
|
methodCalls: [],
|
||
|
inner: inner,
|
||
|
isAjaxError: false,
|
||
|
isSynchronous: false,
|
||
|
url: ''
|
||
|
};
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* We build the methods for the fake XMLHttpRequest.
|
||
|
*/
|
||
|
|
||
|
var copyIn = function() {
|
||
|
copyRequestProperties( this, this.__.inner );
|
||
|
}
|
||
|
var copyOut = function() {
|
||
|
copyRequestProperties( this.__.inner, this, true, this.__.isSynchronous && this.__.isAjaxError );
|
||
|
}
|
||
|
var addHeader = function() {
|
||
|
this.__.inner.setRequestHeader( 'HTTP_X_REQUESTED_WITH', 'XMLHttpRequest' );
|
||
|
}
|
||
|
var isSynchronous = function( args ) {
|
||
|
this.__.isSynchronous = ( args[2] === false );
|
||
|
}
|
||
|
var saveRequest = function( args, method ) {
|
||
|
this.__.methodCalls.push({
|
||
|
method: method,
|
||
|
args: args
|
||
|
});
|
||
|
}
|
||
|
var grabOpen = function( args, method ) {
|
||
|
this.__.url = args[1];
|
||
|
}
|
||
|
|
||
|
wrapMethod( XMLHttpRequest, old, 'open' , saveRequest, copyIn, isSynchronous, grabOpen );
|
||
|
wrapMethod( XMLHttpRequest, old, 'abort' , saveRequest, copyIn );
|
||
|
wrapMethod( XMLHttpRequest, old, 'send' , saveRequest, copyIn, addHeader );
|
||
|
wrapMethod( XMLHttpRequest, old, 'sendAsBinary', saveRequest, copyIn, addHeader );
|
||
|
|
||
|
postMethod( XMLHttpRequest, 'send' , copyOut );
|
||
|
postMethod( XMLHttpRequest, 'sendAsBinary', copyOut );
|
||
|
|
||
|
wrapMethod( XMLHttpRequest, old, 'getAllResponseHeaders', saveRequest );
|
||
|
wrapMethod( XMLHttpRequest, old, 'getResponseHeader' , saveRequest );
|
||
|
wrapMethod( XMLHttpRequest, old, 'setRequestHeader' , saveRequest );
|
||
|
wrapMethod( XMLHttpRequest, old, 'overrideMimeType' , saveRequest );
|
||
|
|
||
|
window.XMLHttpRequest = XMLHttpRequest;
|
||
|
}
|
||
|
})( window );
|
||
|
</script><?php
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* The actual display logic.
|
||
|
* This outputs the error details in HTML.
|
||
|
*/
|
||
|
private function displayError( $message, $errLine, $errFile, $errFileType, $stackTrace, &$fileLinesSets, $numFileLines, $dumpInfo ) {
|
||
|
$applicationRoot = $this->applicationRoot;
|
||
|
$serverName = $this->serverName;
|
||
|
$backgroundText = $this->backgroundText;
|
||
|
$displayLineNumber = $this->displayLineNumber;
|
||
|
$saveUrl = $this->saveUrl;
|
||
|
$isSavingEnabled = $this->isSavingEnabled;
|
||
|
|
||
|
/*
|
||
|
* When a query string is not provided,
|
||
|
* in some versions it's a blank string,
|
||
|
* whilst in others it's not set at all.
|
||
|
*/
|
||
|
if ( isset($_SERVER['QUERY_STRING']) ) {
|
||
|
$requestUrl = str_replace( $_SERVER['QUERY_STRING'], '', $_SERVER['REQUEST_URI'] );
|
||
|
$requestUrlLen = strlen( $requestUrl );
|
||
|
|
||
|
// remove the '?' if it's there (I suspect it isn't always, but don't take my word for it!)
|
||
|
if ( $requestUrlLen > 0 && substr($requestUrl, $requestUrlLen-1) === '?' ) {
|
||
|
$requestUrl = substr( $requestUrl, 0, $requestUrlLen-1 );
|
||
|
}
|
||
|
} else {
|
||
|
$requestUrl = $_SERVER['REQUEST_URI'];
|
||
|
}
|
||
|
|
||
|
header_remove('Content-Transfer-Encoding');
|
||
|
$this->displayHTML(
|
||
|
// pre, in the head
|
||
|
function() use( $message, $errFile, $errLine ) {
|
||
|
echo "<!--\n" .
|
||
|
"$message\n" .
|
||
|
"$errFile, $errLine\n" .
|
||
|
"-->";
|
||
|
},
|
||
|
|
||
|
// the content
|
||
|
function() use (
|
||
|
$requestUrl,
|
||
|
$backgroundText, $serverName, $applicationRoot,
|
||
|
$message, $errLine, $errFile, $errFileType, $stackTrace,
|
||
|
&$fileLinesSets, $numFileLines,
|
||
|
$displayLineNumber,
|
||
|
$dumpInfo,
|
||
|
$isSavingEnabled
|
||
|
) {
|
||
|
if ( $backgroundText ) { ?>
|
||
|
<div id="error-wrap">
|
||
|
<div id="error-back"><?php echo $backgroundText ?></div>
|
||
|
</div>
|
||
|
<?php } ?>
|
||
|
|
||
|
<h2 id="error-file-root"><?php echo $serverName ?> | <?php echo $applicationRoot ?></h2>
|
||
|
<h2 id="ajax-info">
|
||
|
<span id="ajax-tab" class="ajax-button">AJAX PAUSED</span>
|
||
|
|
||
|
<span class="ajax-url"><?php echo $serverName ?><?php echo $requestUrl ?></span>
|
||
|
<span class="ajax-buttons">
|
||
|
<a href="#" id="ajax-close" class="ajax-button">X</a>
|
||
|
<a href="#" id="ajax-retry" class="ajax-button">RETRY</a>
|
||
|
</span>
|
||
|
</h2>
|
||
|
<h1 id="error-title"><?php echo $message ?></h1>
|
||
|
<div class="error-file-top <?php echo ($fileLinesSets ? 'has_code' : '') ?>">
|
||
|
<h2 id="error-file"><span id="error-linenumber"><?php echo $errLine ?></span> <span id="error-filename" class="<?php echo $errFileType ?>"><?php echo $errFile ?></span></h2>
|
||
|
<?php if ( $isSavingEnabled ) { ?>
|
||
|
<a href="#" class="error-file-save">save changes</a>
|
||
|
<?php } ?>
|
||
|
</div>
|
||
|
<?php
|
||
|
|
||
|
if ( $fileLinesSets ) {
|
||
|
?>
|
||
|
<div id="error-editor" class="<?php echo ($displayLineNumber ? '' : 'no-line-nums') ?>">
|
||
|
<noscript>
|
||
|
<div id="noscript-editor">enable JavaScript to view source code</div>
|
||
|
</noscript>
|
||
|
<div id="error-editor-ace"></div>
|
||
|
</div>
|
||
|
<?php
|
||
|
|
||
|
foreach ( $fileLinesSets as $i => $fileLinesSet ) {
|
||
|
$id = $fileLinesSet->getHTMLID();
|
||
|
$fileLines = $fileLinesSet->getLines();
|
||
|
|
||
|
?><div
|
||
|
data-file-id="<?php echo $fileLinesSet->getHTMLID() ?>"
|
||
|
data-file-src="<?php echo $fileLinesSet->getSrc() ?>"
|
||
|
class="error-editor-file"
|
||
|
><?= htmlentities( $fileLinesSet->getContent() ) ?></div><?php
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( $stackTrace !== null ) {
|
||
|
echo $stackTrace;
|
||
|
}
|
||
|
|
||
|
if ( $dumpInfo !== null ) {
|
||
|
echo $dumpInfo;
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Adds:
|
||
|
* = mouse movement for switching the code snippet in real time
|
||
|
*/
|
||
|
function() use ( $saveUrl ) {
|
||
|
?><script>
|
||
|
"use strict";
|
||
|
|
||
|
$(document).ready( function() {
|
||
|
$('#ajax-close', '#ajax-retry').click( function(ev) {
|
||
|
ev.preventDefault();
|
||
|
});
|
||
|
|
||
|
var $editor = $('#error-editor-ace');
|
||
|
if ( $editor.size() > 0 ) {
|
||
|
var editor = null;
|
||
|
var selectFile;
|
||
|
var lines = {};
|
||
|
var lastFileID = null;
|
||
|
|
||
|
var files = $('.error-editor-file');
|
||
|
var EditSession = ace.require('ace/edit_session').EditSession;
|
||
|
var php = ace.require('ace/mode/php');
|
||
|
|
||
|
var changedFiles = {},
|
||
|
fileSessions = {};
|
||
|
var currentFile = null;
|
||
|
|
||
|
var editor = ace.edit( $editor.get(0) );
|
||
|
|
||
|
editor.on('change', function() {
|
||
|
if ( currentFile !== null ) {
|
||
|
changedFiles[currentFile] = true;
|
||
|
}
|
||
|
} );
|
||
|
|
||
|
var selectFile = function( link, line ) {
|
||
|
var fileID = link.attr('data-file-id');
|
||
|
if ( line === undefined ) {
|
||
|
line = link.attr('data-line');
|
||
|
}
|
||
|
|
||
|
setTimeout( function() {
|
||
|
var file;
|
||
|
|
||
|
for ( var i = 0; i < files.size(); i++ ) {
|
||
|
var f = files.get(i);
|
||
|
|
||
|
if ( f.getAttribute('data-file-id') === fileID ) {
|
||
|
file = f;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( file ) {
|
||
|
currentFile = file.getAttribute('data-file-src');
|
||
|
var session;
|
||
|
|
||
|
if ( fileSessions.hasOwnProperty(currentFile) ) {
|
||
|
session = fileSessions[ currentFile ];
|
||
|
} else {
|
||
|
session = fileSessions[ currentFile ] =
|
||
|
new EditSession( file.textContent );
|
||
|
session.setMode( 'ace/mode/php' );
|
||
|
}
|
||
|
|
||
|
editor.setSession( session );
|
||
|
|
||
|
editor.gotoLine( line );
|
||
|
|
||
|
editor.focus();
|
||
|
}
|
||
|
}, 0 );
|
||
|
}
|
||
|
|
||
|
var FADE_SPEED = 150,
|
||
|
lines = $('#error-files .error-file-lines'),
|
||
|
currentID = '#' + lines.filter( '.show' ).attr( 'id' );
|
||
|
|
||
|
var filename = $('#error-filename'),
|
||
|
linenumber = $('#error-linenumber');
|
||
|
|
||
|
$( '.error-stack-trace-line' ).
|
||
|
mouseover( function() {
|
||
|
var $this = $(this);
|
||
|
|
||
|
if ( ! $this.hasClass('select-highlight') ) {
|
||
|
$this.addClass( 'select-highlight' );
|
||
|
}
|
||
|
}).
|
||
|
mouseout( function(ev) {
|
||
|
var $this = $(this);
|
||
|
|
||
|
$this.removeClass( 'select-highlight' );
|
||
|
}).
|
||
|
click( function() {
|
||
|
var $this = $(this);
|
||
|
|
||
|
if ( ! $this.hasClass('highlight') && !$this.hasClass('is-native') ) {
|
||
|
$( '.error-stack-trace-line.highlight' ).removeClass( 'highlight' );
|
||
|
$this.addClass( 'highlight' );
|
||
|
|
||
|
selectFile( $this, $this.attr('data-line') );
|
||
|
}
|
||
|
});
|
||
|
|
||
|
$('#error-stack-trace').mouseleave( function() {
|
||
|
lines.filter('.show').removeClass( 'show' );
|
||
|
lines.filter( currentID ).addClass( 'show' );
|
||
|
});
|
||
|
|
||
|
selectFile( $('.error-stack-trace-line.highlight') );
|
||
|
|
||
|
$('.error-file-save').click( function(ev) {
|
||
|
ev.preventDefault();
|
||
|
ev.stopPropagation();
|
||
|
|
||
|
var files = {},
|
||
|
hasChanges = false;
|
||
|
|
||
|
for ( var k in changedFiles ) {
|
||
|
if ( fileSessions.hasOwnProperty(k) && changedFiles.hasOwnProperty(k) && changedFiles[k] ) {
|
||
|
files[k] = fileSessions[k].getValue();
|
||
|
hasChanges = true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
var retryFun = null;
|
||
|
if ( window.top !== window ) {
|
||
|
var arrFrames = parent.document.getElementsByTagName("IFRAME");
|
||
|
|
||
|
for (var i = 0; i < arrFrames.length; i++) {
|
||
|
if (arrFrames[i].contentWindow === window) {
|
||
|
retryFun = arrFrames[i].__php_error_retry;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( retryFun === null ) {
|
||
|
retryFun = function() {
|
||
|
document.location.reload( true );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( ! hasChanges ) {
|
||
|
retryFun();
|
||
|
} else {
|
||
|
$.ajax({
|
||
|
type: "POST",
|
||
|
url: "<?php echo $saveUrl ?>",
|
||
|
dataType: "json",
|
||
|
|
||
|
data: {
|
||
|
"<?php echo ErrorHandler::POST_FILE_LOCATION ?>": files
|
||
|
},
|
||
|
|
||
|
success: function(res, status, xhr) {
|
||
|
retryFun();
|
||
|
},
|
||
|
|
||
|
beforeSend: function(xhr) {
|
||
|
xhr.setRequestHeader( "<?php echo ErrorHandler::HEADER_SAVE_FILE ?>", 'true' );
|
||
|
}
|
||
|
});
|
||
|
}
|
||
|
} );
|
||
|
}
|
||
|
} );
|
||
|
</script><?php
|
||
|
}
|
||
|
);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* A generic function for clearing the buffer, and displaying error output.
|
||
|
*
|
||
|
* A function needs to be given, and then this is run at the correct time.
|
||
|
* There are two ways this can be used:
|
||
|
*
|
||
|
* displayHTML( $head, $body )
|
||
|
*
|
||
|
* Here 'head' is run straight after the doctype, whilst 'body' is run as
|
||
|
* the body for the content. The other way is:
|
||
|
*
|
||
|
* displayHTML( $body )
|
||
|
*
|
||
|
* Here there is only content.
|
||
|
*/
|
||
|
function displayHTML( Closure $head, $body=null, $javascript=null ) {
|
||
|
if ( func_num_args() === 2 ) {
|
||
|
$body = $head;
|
||
|
$head = null;
|
||
|
}
|
||
|
|
||
|
// clean out anything displayed already
|
||
|
try {
|
||
|
@ob_clean();
|
||
|
} catch ( Exception $ex ) { /* do nothing */ }
|
||
|
|
||
|
if (!$this->htmlOnly && ErrorHandler::isNonPHPRequest()) {
|
||
|
@header( "Content-Type: text/html" );
|
||
|
}
|
||
|
@header( ErrorHandler::PHP_ERROR_MAGIC_HEADER_KEY . ': ' . ErrorHandler::PHP_ERROR_MAGIC_HEADER_VALUE );
|
||
|
|
||
|
echo '<!DOCTYPE html>';
|
||
|
|
||
|
if ( $head !== null ) {
|
||
|
$head();
|
||
|
}
|
||
|
|
||
|
echo "<link href='http://fonts.googleapis.com/css?family=Droid+Sans+Mono' rel='stylesheet' type='text/css'>";
|
||
|
|
||
|
?><style>
|
||
|
html, body {
|
||
|
margin: 0;
|
||
|
padding: 0;
|
||
|
width: 100%;
|
||
|
height: 100%;
|
||
|
}
|
||
|
body {
|
||
|
color: #f0f0f0;
|
||
|
|
||
|
tab-size: 4;
|
||
|
}
|
||
|
|
||
|
::-moz-selection{background: #662039 !important; color: #fff !important; text-shadow: none;}
|
||
|
::selection {background: #662039 !important; color: #fff !important; text-shadow: none;}
|
||
|
|
||
|
a,
|
||
|
.error-stack-trace-line {
|
||
|
-webkit-transition: color 120ms linear, background 120ms linear;
|
||
|
-moz-transition: color 120ms linear, background 120ms linear;
|
||
|
-ms-transition: color 120ms linear, background 120ms linear;
|
||
|
-o-transition: color 120ms linear, background 120ms linear;
|
||
|
transition: color 120ms linear, background 120ms linear;
|
||
|
}
|
||
|
|
||
|
a,
|
||
|
a:visited,
|
||
|
a:hover,
|
||
|
a:active {
|
||
|
color: #9ae;
|
||
|
text-decoration: none;
|
||
|
}
|
||
|
a:hover,
|
||
|
a.error-editor-link:hover {
|
||
|
color: #aff;
|
||
|
}
|
||
|
a.error-editor-link,
|
||
|
a.error-editor-link:visited,
|
||
|
a.error-editor-link:active {
|
||
|
color: inherit;
|
||
|
}
|
||
|
|
||
|
h2,
|
||
|
#error-editor .ace_line,
|
||
|
#error-editor .ace_editor,
|
||
|
#error-editor .ace_gutter,
|
||
|
.background {
|
||
|
font-size: 16px;
|
||
|
font-family: inconsolata, 'Droid Sans Mono', "DejaVu Sans Mono", consolas, monospace;
|
||
|
}
|
||
|
#error-editor .ace_line,
|
||
|
#error-editor .ace_editor,
|
||
|
#error-editor .ace_gutter,
|
||
|
.background {
|
||
|
line-height: 18px;
|
||
|
}
|
||
|
#error-editor.no-line-nums .ace_gutter {
|
||
|
display: none;
|
||
|
}
|
||
|
|
||
|
h1,
|
||
|
h2 {
|
||
|
font-family: "Segoe UI Light","Helvetica Neue",'RobotoLight',"Segoe UI","Segoe WP",sans-serif;
|
||
|
font-weight: 100;
|
||
|
line-height: normal;
|
||
|
}
|
||
|
h1 {
|
||
|
font-size: 42px;
|
||
|
margin-bottom: 0;
|
||
|
}
|
||
|
h2 {
|
||
|
font-size: 28px;
|
||
|
margin-top: 0;
|
||
|
}
|
||
|
.background {
|
||
|
width: 100%;
|
||
|
background: #111;
|
||
|
|
||
|
padding: 18px 24px;
|
||
|
-moz-box-sizing: border-box;
|
||
|
box-sizing: border-box;
|
||
|
|
||
|
/*
|
||
|
* Take over the page via CSS,
|
||
|
* so we block anything already rendered.
|
||
|
*/
|
||
|
position: fixed;
|
||
|
top: 0;
|
||
|
left: 0;
|
||
|
right: 0;
|
||
|
bottom: 0;
|
||
|
|
||
|
z-index: 100000;
|
||
|
|
||
|
height: 100%;
|
||
|
overflow: auto;
|
||
|
}
|
||
|
html.ajax {
|
||
|
background: transparent;
|
||
|
}
|
||
|
html.ajax > body {
|
||
|
background: rgba( 0, 0, 0, 0.3 );
|
||
|
-moz-box-sizing: border-box;
|
||
|
box-sizing: border-box;
|
||
|
|
||
|
padding: 30px 48px;
|
||
|
}
|
||
|
html.ajax > body > .background {
|
||
|
border-radius: 4px;
|
||
|
box-shadow: 5px 8px 18px rgba( 0, 0, 0, 0.4 );
|
||
|
|
||
|
height: auto;
|
||
|
min-height: 0;
|
||
|
|
||
|
overflow: hidden;
|
||
|
|
||
|
position: relative;
|
||
|
top: auto;
|
||
|
left: auto;
|
||
|
right: auto;
|
||
|
bottom: auto;
|
||
|
}
|
||
|
|
||
|
#ajax-info,
|
||
|
.ajax-button {
|
||
|
font-size: 26px;
|
||
|
}
|
||
|
#ajax-info {
|
||
|
display: none;
|
||
|
position: relative;
|
||
|
line-height: 100%;
|
||
|
|
||
|
white-space: nowrap;
|
||
|
}
|
||
|
html.ajax #ajax-info {
|
||
|
display: block;
|
||
|
}
|
||
|
html.ajax #error-file-root {
|
||
|
display: none;
|
||
|
}
|
||
|
.ajax-button {
|
||
|
padding: 2px 12px 5px 12px;
|
||
|
margin-top: -3px;
|
||
|
border-radius: 3px;
|
||
|
color: #bbb;
|
||
|
font-weight: 400;
|
||
|
}
|
||
|
.ajax-button,
|
||
|
.ajax-button:visited,
|
||
|
.ajax-button:active,
|
||
|
.ajax-button:hover {
|
||
|
text-decoration: none;
|
||
|
}
|
||
|
a.ajax-button:hover {
|
||
|
color: #fff;
|
||
|
}
|
||
|
#ajax-tab {
|
||
|
float: left;
|
||
|
margin-right: 12px;
|
||
|
|
||
|
background: #000;
|
||
|
color: inherit;
|
||
|
border: 1px solid #333;
|
||
|
box-shadow: 0 0 2px #222;
|
||
|
margin-top: -4px;
|
||
|
}
|
||
|
.ajax-buttons {
|
||
|
position: absolute;
|
||
|
right: 0;
|
||
|
top: 0;
|
||
|
}
|
||
|
#ajax-retry {
|
||
|
float: right;
|
||
|
background: #0E4973;
|
||
|
|
||
|
margin-right: 12px;
|
||
|
}
|
||
|
#ajax-retry:hover {
|
||
|
background: #0C70B7;
|
||
|
}
|
||
|
#ajax-close {
|
||
|
float: right;
|
||
|
background: #622;
|
||
|
}
|
||
|
#ajax-close:hover {
|
||
|
background: #aa4040;
|
||
|
}
|
||
|
|
||
|
#error-title {
|
||
|
margin-top: 6px;
|
||
|
position: relative;
|
||
|
white-space: pre-wrap;
|
||
|
}
|
||
|
|
||
|
<?php
|
||
|
/*
|
||
|
* Error Background Text.
|
||
|
*/
|
||
|
?>
|
||
|
#error-wrap {
|
||
|
right: 0;
|
||
|
top: 0;
|
||
|
position: absolute;
|
||
|
|
||
|
width : 100%;
|
||
|
height: 0;
|
||
|
}
|
||
|
#error-back {
|
||
|
font-size: 240px;
|
||
|
color: #211600;
|
||
|
position: absolute;
|
||
|
top: 60px;
|
||
|
right: -40px;
|
||
|
|
||
|
-webkit-transform: rotate( 24deg );
|
||
|
-moz-transform: rotate( 24deg );
|
||
|
-ms-transform: rotate( 24deg );
|
||
|
-o-transform: rotate( 24deg );
|
||
|
transform: rotate( 24deg );
|
||
|
}
|
||
|
|
||
|
<?php
|
||
|
/*
|
||
|
* Code Snippets at the top
|
||
|
*/
|
||
|
?>
|
||
|
.error-file-top.has_code {
|
||
|
position: relative;
|
||
|
height: 42px;
|
||
|
margin: 16px 0 3px 0;
|
||
|
}
|
||
|
.error-file-top > h2 {
|
||
|
position: absolute;
|
||
|
|
||
|
left: 0;
|
||
|
right: 129px;
|
||
|
bottom: 0;
|
||
|
|
||
|
margin: 0;
|
||
|
}
|
||
|
.error-file-top.has_code > h2 {
|
||
|
bottom: 3px;
|
||
|
left: 167px;
|
||
|
}
|
||
|
|
||
|
.error-file-save {
|
||
|
position: absolute;
|
||
|
right: 0;
|
||
|
bottom: 0;
|
||
|
width: 160px;
|
||
|
line-height: 36px;
|
||
|
|
||
|
text-align: center;
|
||
|
|
||
|
color: #555;
|
||
|
border: 1px solid #555;
|
||
|
|
||
|
border-radius: 3px;
|
||
|
|
||
|
-webkit-transition: color 200ms linear, border-color 200ms linear;
|
||
|
-moz-transition: color 200ms linear, border-color 200ms linear;
|
||
|
transition: color 200ms linear, border-color 200ms linear;
|
||
|
}
|
||
|
.error-file-save,
|
||
|
.error-file-save:active,
|
||
|
.error-file-save:visited,
|
||
|
.error-file-save:hover {
|
||
|
text-decoration: none;
|
||
|
|
||
|
color: #555;
|
||
|
border-color: #555;
|
||
|
}
|
||
|
.error-file-save:hover {
|
||
|
color: #fff;
|
||
|
border-color: #fff;
|
||
|
}
|
||
|
|
||
|
#error-linenumber {
|
||
|
position: absolute;
|
||
|
text-align: right;
|
||
|
right: 101%;
|
||
|
width: 178px;
|
||
|
}
|
||
|
#ajax-info,
|
||
|
#error-file-root {
|
||
|
color: #666;
|
||
|
}
|
||
|
#error-file-root {
|
||
|
position: relative;
|
||
|
}
|
||
|
#error-files {
|
||
|
line-height: 0;
|
||
|
font-size: 0;
|
||
|
|
||
|
position: relative;
|
||
|
padding: 9px 0 36px 0;
|
||
|
|
||
|
display: inline-block;
|
||
|
|
||
|
width: 100%;
|
||
|
-moz-box-sizing: border-box;
|
||
|
box-sizing: border-box;
|
||
|
padding-left: 166px;
|
||
|
|
||
|
overflow: hidden;
|
||
|
}
|
||
|
/**
|
||
|
* Two transitions are used to get them to smoothly fade,
|
||
|
* in both directions.
|
||
|
*
|
||
|
* The second keeps it on screen for long enough for the
|
||
|
* fade to occur, and then does the margin transtion to move
|
||
|
* it out.
|
||
|
*/
|
||
|
.error-file-lines {
|
||
|
display: inline-block;
|
||
|
opacity: 0;
|
||
|
|
||
|
float: left;
|
||
|
clear: none;
|
||
|
|
||
|
width: 100%;
|
||
|
margin-right: -100%;
|
||
|
|
||
|
-webkit-transition: opacity 300ms;
|
||
|
-moz-transition: opacity 300ms;
|
||
|
-ms-transition: opacity 300ms;
|
||
|
-o-transition: opacity 300ms;
|
||
|
transition: opacity 300ms;
|
||
|
}
|
||
|
.error-file-lines.show {
|
||
|
height: auto;
|
||
|
|
||
|
opacity: 1;
|
||
|
|
||
|
margin: 0;
|
||
|
|
||
|
-webkit-transition: opacity 300ms, margin 100ms linear 300ms;
|
||
|
-moz-transition: opacity 300ms, margin 100ms linear 300ms;
|
||
|
-ms-transition: opacity 300ms, margin 100ms linear 300ms;
|
||
|
-o-transition: opacity 300ms, margin 100ms linear 300ms;
|
||
|
transition: opacity 300ms, margin 100ms linear 300ms;
|
||
|
}
|
||
|
.error-file-line {
|
||
|
line-height: 21px;
|
||
|
|
||
|
font-size: 16px;
|
||
|
|
||
|
color: #ddd;
|
||
|
list-style-type: none;
|
||
|
|
||
|
/* needed for empty lines */
|
||
|
min-height: 20px;
|
||
|
padding-right: 18px;
|
||
|
padding-bottom: 1px;
|
||
|
|
||
|
box-sizing: border-box;
|
||
|
padding-left: 166px;
|
||
|
|
||
|
border-radius: 2px;
|
||
|
|
||
|
display: inline-block;
|
||
|
float: left;
|
||
|
clear: both;
|
||
|
|
||
|
position: relative;
|
||
|
|
||
|
/* Chrome fix */
|
||
|
min-width: 50%;
|
||
|
}
|
||
|
.error-file-line-number {
|
||
|
position: absolute;
|
||
|
top: 0;
|
||
|
right: 100%;
|
||
|
margin-right: 12px;
|
||
|
display: block;
|
||
|
text-indent: 0;
|
||
|
text-align: left;
|
||
|
}
|
||
|
|
||
|
#error-editor {
|
||
|
width: 100%;
|
||
|
|
||
|
position: relative;
|
||
|
|
||
|
margin: 0 0 36px 0;
|
||
|
}
|
||
|
#error-editor {
|
||
|
height: 450px;
|
||
|
}
|
||
|
#noscript-editor {
|
||
|
width: 100%;
|
||
|
line-height: 400px;
|
||
|
font-size: 32px;
|
||
|
text-align: center;
|
||
|
}
|
||
|
#error-editor-ace {
|
||
|
top: 0;
|
||
|
bottom: 0;
|
||
|
left : 0;
|
||
|
right: 0;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor > .ace_gutter {
|
||
|
background: transparent;
|
||
|
color: #555;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_print_margin_layer {
|
||
|
display: none;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_indent-guide {
|
||
|
background: none;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_scroller {
|
||
|
background-color: #111;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_text-layer {
|
||
|
color: #F8F8F8;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_cursor {
|
||
|
border-left: 2px solid #A7A7A7;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_cursor.ace_overwrite {
|
||
|
border-left: 0px;
|
||
|
border-bottom: 1px solid #A7A7A7;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_marker-layer .ace_selection {
|
||
|
background: rgba(221, 240, 255, 0.20);
|
||
|
}
|
||
|
#error-editor-ace.ace_editor.multiselect .ace_selection.start {
|
||
|
box-shadow: 0 0 3px 0px #141414;
|
||
|
border-radius: 2px;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_marker-layer .ace_step {
|
||
|
background: rgb(102, 82, 0);
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_marker-layer .ace_bracket {
|
||
|
margin: -1px 0 0 -1px;
|
||
|
border: 1px solid rgba(255, 255, 255, 0.25);
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_marker-layer .ace_active_line {
|
||
|
background: rgba(255, 255, 255, 0.031);
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_gutter_active_line {
|
||
|
background-color: rgba(255, 255, 255, 0.031);
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_marker-layer .ace_selected_word {
|
||
|
border: 1px solid rgba(221, 240, 255, 0.20);
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_invisible {
|
||
|
color: rgba(255, 255, 255, 0.25);
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_identifier {
|
||
|
color: #F9EE98;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_keyword,
|
||
|
#error-editor-ace.ace_editor .ace_meta {
|
||
|
color:#C07041;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_constant,
|
||
|
#error-editor-ace.ace_editor .ace_constant.ace_other {
|
||
|
color:#cF5d33;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_constant.ace_character, {
|
||
|
color:#CF6A4C;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_constant.ace_character.ace_escape, {
|
||
|
color:#CF6A4C;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_invalid.ace_illegal {
|
||
|
color:#F8F8F8;
|
||
|
background-color:rgba(86, 45, 86, 0.75);
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_invalid.ace_deprecated {
|
||
|
text-decoration:underline;
|
||
|
font-style:italic;
|
||
|
color:#D2A8A1;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_support {
|
||
|
color:#9B859D;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_support.ace_constant {
|
||
|
color:#CF6A4C;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_fold {
|
||
|
background-color: #AC885B;
|
||
|
border-color: #F8F8F8;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_support.ace_function {
|
||
|
color:#DAD085;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_storage {
|
||
|
color:#F9EE98;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_variable {
|
||
|
color:#AC885B;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_string {
|
||
|
color:#7C9D5D;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_string.ace_regexp {
|
||
|
color:#E9C062;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_comment {
|
||
|
font-style:italic;
|
||
|
color:#5F5A60;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_variable {
|
||
|
color:#798aA0;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_xml_pe {
|
||
|
color:#494949;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_meta.ace_tag {
|
||
|
color:#AC885B;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_entity.ace_name.ace_function {
|
||
|
color:#AC885B;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_markup.ace_underline {
|
||
|
text-decoration:underline;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_markup.ace_heading {
|
||
|
color:#CF6A4C;
|
||
|
}
|
||
|
#error-editor-ace.ace_editor .ace_markup.ace_list {
|
||
|
color:#F9EE98;
|
||
|
}
|
||
|
.ace_sb::-webkit-scrollbar {
|
||
|
background: #111;
|
||
|
border: 1px solid #333;
|
||
|
border-radius: 2px;
|
||
|
}
|
||
|
.ace_sb::-webkit-scrollbar-thumb {
|
||
|
background: #333;
|
||
|
}
|
||
|
.ace_sb::-webkit-scrollbar-corner {
|
||
|
width: 0;
|
||
|
height: 0;
|
||
|
}
|
||
|
|
||
|
.error-editor-file {
|
||
|
display: none;
|
||
|
}
|
||
|
<?php
|
||
|
/*
|
||
|
* Stack Trace
|
||
|
*/
|
||
|
?>
|
||
|
#error-stack-trace,
|
||
|
.error-stack-trace-line {
|
||
|
border-spacing: 0;
|
||
|
width: 100%;
|
||
|
}
|
||
|
#error-stack-trace {
|
||
|
position: relative;
|
||
|
|
||
|
line-height: 28px;
|
||
|
cursor: pointer;
|
||
|
}
|
||
|
.error-stack-trace-exception {
|
||
|
color: #b33;
|
||
|
}
|
||
|
.error-stack-trace-exception > td {
|
||
|
padding-top: 18px;
|
||
|
}
|
||
|
.error-stack-trace-line {
|
||
|
float: left;
|
||
|
}
|
||
|
.error-stack-trace-line.is-exception {
|
||
|
margin-top: 18px;
|
||
|
border-top: 1px solid #422;
|
||
|
}
|
||
|
.error-stack-trace-line:first-of-type > td:first-of-type {
|
||
|
border-top-left-radius: 2px;
|
||
|
}
|
||
|
.error-stack-trace-line:first-of-type > td:last-of-type {
|
||
|
border-top-right-radius: 2px;
|
||
|
}
|
||
|
.error-stack-trace-line:last-of-type > td:first-of-type {
|
||
|
border-bottom-left-radius: 2px;
|
||
|
}
|
||
|
.error-stack-trace-line:last-of-type > td:last-of-type {
|
||
|
border-bottom-right-radius: 2px;
|
||
|
}
|
||
|
|
||
|
.error-stack-trace-line > td {
|
||
|
padding: 3px 0;
|
||
|
vertical-align: top;
|
||
|
}
|
||
|
.error-stack-trace-line > .linenumber,
|
||
|
.error-stack-trace-line > .filename,
|
||
|
.error-stack-trace-line > .file-internal-php,
|
||
|
.error-stack-trace-line > .lineinfo {
|
||
|
padding-left: 18px;
|
||
|
padding-right: 12px;
|
||
|
}
|
||
|
.error-stack-trace-line > .linenumber,
|
||
|
.error-stack-trace-line > .file-internal-php,
|
||
|
.error-stack-trace-line > .filename {
|
||
|
white-space: pre;
|
||
|
}
|
||
|
.error-stack-trace-line > .linenumber {
|
||
|
text-align: right;
|
||
|
}
|
||
|
.error-stack-trace-line > .file-internal-php,
|
||
|
.error-stack-trace-line > .filename {
|
||
|
}
|
||
|
.error-stack-trace-line > .lineinfo {
|
||
|
width: 100%; /* fix for chrome */
|
||
|
padding-right:18px;
|
||
|
padding-left: 82px;
|
||
|
text-indent: -64px;
|
||
|
}
|
||
|
<?php
|
||
|
/*
|
||
|
* Error Dump Info (post, get, session)
|
||
|
*/
|
||
|
?>
|
||
|
.error-dumps {
|
||
|
position: relative;
|
||
|
|
||
|
margin-top: 48px;
|
||
|
padding-top: 32px;
|
||
|
width: 100%;
|
||
|
max-width: 100%;
|
||
|
overflow: hidden;
|
||
|
}
|
||
|
.error_dump {
|
||
|
float: left;
|
||
|
clear: none;
|
||
|
|
||
|
-moz-box-sizing: border-box;
|
||
|
box-sizing: border-box;
|
||
|
|
||
|
padding: 0 32px 24px 12px;
|
||
|
max-width: 100%;
|
||
|
}
|
||
|
.error_dump.dump_request {
|
||
|
clear: left;
|
||
|
max-width: 50%;
|
||
|
min-width: 600px;
|
||
|
}
|
||
|
.error_dump.dump_response {
|
||
|
max-width: 50%;
|
||
|
min-width: 600px;
|
||
|
}
|
||
|
.error_dump.dump_server {
|
||
|
width: 100%;
|
||
|
clear: both;
|
||
|
}
|
||
|
.error_dump_header {
|
||
|
color: #eb4;
|
||
|
margin: 0;
|
||
|
margin-left: -6px;
|
||
|
}
|
||
|
.error_dump_key,
|
||
|
.error_dump_mapping,
|
||
|
.error_dump_value {
|
||
|
white-space: pre;
|
||
|
padding: 3px 6px 3px 6px;
|
||
|
float: left;
|
||
|
}
|
||
|
.error_dump_key {
|
||
|
clear: left;
|
||
|
}
|
||
|
.error_dump_mapping {
|
||
|
padding: 3px 12px;
|
||
|
}
|
||
|
.error_dump_value {
|
||
|
clear: right;
|
||
|
white-space: normal;
|
||
|
max-width: 100%;
|
||
|
}
|
||
|
|
||
|
<?php
|
||
|
/*
|
||
|
* Code and Stack highlighting colours
|
||
|
*
|
||
|
* The way this works, is that syntax highlighting is turned off
|
||
|
* for .pre-highlight. It then gets turns on for .pre-highlight,
|
||
|
* if it matches certain criteria.
|
||
|
*
|
||
|
* The emphasis is that pre-highlight is by default 'no highlight'.
|
||
|
*/
|
||
|
?>
|
||
|
.pre-highlight,
|
||
|
.highlight {
|
||
|
}
|
||
|
.is-native,
|
||
|
.pre-highlight {
|
||
|
opacity: 0.3;
|
||
|
color: #999;
|
||
|
}
|
||
|
.is-native {
|
||
|
opacity: 0.3 !important;
|
||
|
}
|
||
|
.highlight,
|
||
|
.pre-highlight.highlight,
|
||
|
.highlight ~ .pre-highlight {
|
||
|
color: #eee;
|
||
|
opacity: 1;
|
||
|
}
|
||
|
|
||
|
.select-highlight {
|
||
|
background: #261313;
|
||
|
}
|
||
|
.select-highlight.is-native {
|
||
|
background: #222;
|
||
|
}
|
||
|
.highlight {
|
||
|
background: #391414;
|
||
|
}
|
||
|
.highlight.select-highlight {
|
||
|
background: #451915;
|
||
|
}
|
||
|
|
||
|
.pre-highlight span,
|
||
|
.pre-highlight:not(.highlight):first-of-type span {
|
||
|
color : #999;
|
||
|
border: none !important;
|
||
|
}
|
||
|
|
||
|
<?php
|
||
|
/*
|
||
|
* Syntax Highlighting
|
||
|
*/
|
||
|
?>
|
||
|
.pre-highlight:first-of-type .syntax-class,
|
||
|
.highlight ~ .pre-highlight .syntax-class,
|
||
|
.pre-highlight.highlight .syntax-class,
|
||
|
.syntax-class {
|
||
|
color: #C07041;
|
||
|
}
|
||
|
.pre-highlight:first-of-type .syntax-function,
|
||
|
.highlight ~ .pre-highlight .syntax-function,
|
||
|
.pre-highlight.highlight .syntax-function,
|
||
|
.syntax-function {
|
||
|
color: #F9EE98;
|
||
|
}
|
||
|
.pre-highlight:first-of-type .syntax-literal,
|
||
|
.highlight ~ .pre-highlight .syntax-literal,
|
||
|
.pre-highlight.highlight .syntax-literal,
|
||
|
.syntax-literal {
|
||
|
color: #cF5d33;
|
||
|
}
|
||
|
.pre-highlight:first-of-type .syntax-string,
|
||
|
.highlight ~ .pre-highlight .syntax-string,
|
||
|
.pre-highlight.highlight .syntax-string,
|
||
|
.syntax-string {
|
||
|
color: #7C9D5D;
|
||
|
}
|
||
|
.pre-highlight:first-of-type .syntax-variable-not-important,
|
||
|
.highlight ~ .pre-highlight .syntax-variable-not-important,
|
||
|
.pre-highlight.highlight .syntax-variable-not-important,
|
||
|
.syntax-variable-not-important {
|
||
|
opacity: 0.5;
|
||
|
}
|
||
|
.pre-highlight:first-of-type .syntax-higlight-variable,
|
||
|
.highlight ~ .pre-highlight .syntax-higlight-variable,
|
||
|
.pre-highlight.highlight .syntax-higlight-variable,
|
||
|
.syntax-higlight-variable {
|
||
|
color: #f00;
|
||
|
border-bottom: 3px dashed #c33;
|
||
|
}
|
||
|
.pre-highlight:first-of-type .syntax-variable,
|
||
|
.highlight ~ .pre-highlight .syntax-variable,
|
||
|
.pre-highlight.highlight .syntax-variable,
|
||
|
.syntax-variable {
|
||
|
color: #798aA0;
|
||
|
}
|
||
|
.pre-highlight:first-of-type .syntax-keyword,
|
||
|
.highlight ~ .pre-highlight .syntax-keyword,
|
||
|
.pre-highlight.highlight .syntax-keyword,
|
||
|
.syntax-keyword {
|
||
|
color: #C07041;
|
||
|
}
|
||
|
.pre-highlight:first-of-type .syntax-comment,
|
||
|
.highlight ~ .pre-highlight .syntax-comment,
|
||
|
.pre-highlight.highlight .syntax-comment,
|
||
|
.syntax-comment {
|
||
|
color: #5a5a5a;
|
||
|
}
|
||
|
|
||
|
<?php
|
||
|
/*
|
||
|
* File Highlighting
|
||
|
*/
|
||
|
?>
|
||
|
.file-internal-php {
|
||
|
color: #555 !important;
|
||
|
}
|
||
|
.pre-highlight:first-of-type .file-common,
|
||
|
.highlight ~ .pre-highlight .file-common,
|
||
|
.pre-highlight.highlight .file-common,
|
||
|
.file-common {
|
||
|
color: #eb4;
|
||
|
}
|
||
|
.pre-highlight:first-of-type .file-ignore,
|
||
|
.highlight ~ .pre-highlight .file-ignore,
|
||
|
.pre-highlight.highlight .file-ignore,
|
||
|
.file-ignore {
|
||
|
color: #585;
|
||
|
}
|
||
|
.pre-highlight:first-of-type .file-app,
|
||
|
.highlight ~ .pre-highlight .file-app,
|
||
|
.pre-highlight.highlight .file-app,
|
||
|
.file-app {
|
||
|
color: #66c6d5;
|
||
|
}
|
||
|
.pre-highlight:first-of-type .file-root,
|
||
|
.highlight ~ .pre-highlight .file-root,
|
||
|
.pre-highlight.highlight .file-root,
|
||
|
.file-root {
|
||
|
color: #b69;
|
||
|
}
|
||
|
</style><?php
|
||
|
|
||
|
?><div class="background"><?php
|
||
|
$body();
|
||
|
?></div><?php
|
||
|
|
||
|
/*
|
||
|
* ace.ajax
|
||
|
*/
|
||
|
?><script>
|
||
|
(function(){function h(a){var c=function(a,b){return e("",a,b)},f=b;a&&(b[a]||(b[a]={}),f=b[a]);if(!f.define||!f.define.packaged)d.original=f.define,f.define=d,f.define.packaged=!0;if(!f.require||!f.require.packaged)e.original=f.require,f.require=c,f.require.packaged=!0}var a="ace",b=function(){return this}();if(!a&&typeof requirejs!="undefined"){var c=b.define;b.define=function(a,b,d){return typeof d!="function"?c.apply(this,arguments):ace.define(a,b,function(a,c,e){return b[2]=="module"&&(e.packaged=!0),d.apply(this,arguments)})},b.define.packaged=!0;return}var d=function(a,b,c){if(typeof a!="string"){d.original?d.original.apply(window,arguments):(console.error("dropping module because define wasn't a string."),console.trace());return}arguments.length==2&&(c=b),d.modules||(d.modules={}),d.modules[a]=c},e=function(a,b,c){if(Object.prototype.toString.call(b)==="[object Array]"){var d=[];for(var f=0,h=b.length;f<h;++f){var i=g(a,b[f]);if(!i&&e.original)return e.original.apply(window,arguments);d.push(i)}c&&c.apply(null,d)}else{if(typeof b=="string"){var j=g(a,b);return!j&&e.original?e.original.apply(window,arguments):(c&&c(),j)}if(e.original)return e.original.apply(window,arguments)}},f=function(a,b){if(b.indexOf("!")!==-1){var c=b.split("!");return f(a,c[0])+"!"+f(a,c[1])}if(b.charAt(0)=="."){var d=a.split("/").slice(0,-1).join("/");b=d+"/"+b;while(b.indexOf(".")!==-1&&e!=b){var e=b;b=b.replace(/\/\.\//,"/").replace(/[^\/]+\/\.\.\//,"")}}return b},g=function(a,b){b=f(a,b);var c=d.modules[b];if(!c)return null;if(typeof c=="function"){var g={},h={id:b,uri:"",exports:g,packaged:!0},i=function(a,c){return e(b,a,c)},j=c(i,g,h);return g=j||h.exports,d.modules[b]=g,g}return c};h(a)})(),ace.define("ace/requirejs/text",["require","exports","module"],function(a,b,c){b.load=function(b,c,d,e){a("ace/lib/net").get(c.toUrl(b),d)}}),ace.define("ace/ace",["require","exports","module","ace/lib/fixoldbrowsers","ace/lib/dom","ace/lib/event","ace/editor","ace/edit_session","ace/undomanager","ace/virtual_renderer","ace/multi_select","ace/worker/worker_client","ace/keyboard/hash_handler","ace/keyboard/state_handler","ace/placeholder","ace/config","ace/theme/textmate"],function(a,b,c){a("./lib/fixoldbrowsers");var d=a("./lib/dom"),e=a("./lib/event"),f=a("./editor").Editor,g=a("./edit_session").EditSession,h=a("./undomanager").UndoManager,i=a("./virtual_renderer").VirtualRenderer,j=a("./multi_select").MultiSelect;a("./worker/worker_client"),a("./keyboard/hash_handler"),a("./keyboard/state_handler"),a("./placeholder"),b.config=a("./config"),b.edit=function(b){typeof b=="string"&&(b=document.getElementById(b));if(b.env&&b.env.editor instanceof f)return b.env.editor;var c=new g(d.getInnerText(b));c.setUndoManager(new h),b.innerHTML="";var k=new f(new i(b,a("./theme/textmate")));new j(k),k.setSession(c);var l={};return l.document=c,l.editor=k,k.resize(),e.addListener(window,"resize",function(){k.resize()}),b.env=l,k.env=l,k}}),ace.define("ace/lib/fixoldbrowsers",["require","exports","module","ace/lib/regexp","ace/lib/es5-shim"],function(a,b,c){a("./regexp"),a("./es5-shim")}),ace.define("ace/lib/regexp",["require","exports","module"],function(a,b,c){function g(a){return(a.global?"g":"")+(a.ignoreCase?"i":"")+(a.multiline?"m":"")+(a.extended?"x":"")+(a.sticky?"y":"")}function h(a,b,c){if(Array.prototype.indexOf)return a.indexOf(b,c);for(var d=c||0;d<a.length;d++)if(a[d]===b)return d;return-1}var d={exec:RegExp.prototype.exec,test:RegExp.prototype.test,match:String.prototype.match,replace:String.prototype.replace,split:String.prototype.split},e=d.exec.call(/()??/,"")[1]===undefined,f=function(){var a=/^/g;return d.test.call(a,""),!a.lastIndex}();if(f&&e)return;RegExp.prototype.exec=function(a){var b=d.exec.apply(this,arguments),c,i;if(typeof a=="string"&&b){!e&&b.length>1&&h(b,"")>-1&&(i=RegExp(this.source,d.replace.call(g(this),"g","")),d.replace.call(a.slice(b.index),i,function(){for(var a=1;a<arguments.length-2;a++)arguments[a]===undefined&&(b[a]=undefined)}));if(this._xregexp&&this._xregexp.captureNames)for(var j=1;j<b.length;
|
||
|
(function() {
|
||
|
ace.require(["ace/ace"], function(a) {
|
||
|
a && a.config.init();
|
||
|
if (!window.ace)
|
||
|
window.ace = {};
|
||
|
for (var key in a) if (a.hasOwnProperty(key))
|
||
|
ace[key] = a[key];
|
||
|
});
|
||
|
})();
|
||
|
|
||
|
ace.define("ace/mode/php",["require","exports","module","ace/lib/oop","ace/mode/text","ace/tokenizer","ace/mode/php_highlight_rules","ace/mode/matching_brace_outdent","ace/range","ace/mode/behaviour/cstyle","ace/mode/folding/cstyle"],function(a,b,c){var d=a("../lib/oop"),e=a("./text").Mode,f=a("../tokenizer").Tokenizer,g=a("./php_highlight_rules").PhpHighlightRules,h=a("./matching_brace_outdent").MatchingBraceOutdent,i=a("../range").Range,j=a("./behaviour/cstyle").CstyleBehaviour,k=a("./folding/cstyle").FoldMode,l=function(){this.$tokenizer=new f((new g).getRules()),this.$outdent=new h,this.$behaviour=new j,this.foldingRules=new k};d.inherits(l,e),function(){this.toggleCommentLines=function(a,b,c,d){var e=!0,f=/^(\s*)#/;for(var g=c;g<=d;g++)if(!f.test(b.getLine(g))){e=!1;break}if(e){var h=new i(0,0,0,0);for(var g=c;g<=d;g++){var j=b.getLine(g),k=j.match(f);h.start.row=g,h.end.row=g,h.end.column=k[0].length,b.replace(h,k[1])}}else b.indentRows(c,d,"#")},this.getNextLineIndent=function(a,b,c){var d=this.$getIndent(b),e=this.$tokenizer.getLineTokens(b,a),f=e.tokens;if(f.length&&f[f.length-1].type=="comment")return d;if(a=="start"){var g=b.match(/^.*[\{\(\[\:]\s*$/);g&&(d+=c)}return d},this.checkOutdent=function(a,b,c){return this.$outdent.checkOutdent(b,c)},this.autoOutdent=function(a,b,c){this.$outdent.autoOutdent(b,c)}}.call(l.prototype),b.Mode=l}),ace.define("ace/mode/php_highlight_rules",["require","exports","module","ace/lib/oop","ace/lib/lang","ace/mode/doc_comment_highlight_rules","ace/mode/text_highlight_rules"],function(a,b,c){var d=a("../lib/oop"),e=a("../lib/lang"),f=a("./doc_comment_highlight_rules").DocCommentHighlightRules,g=a("./text_highlight_rules").TextHighlightRules,h=function(){var a=f,b=e.arrayToMap("abs|acos|acosh|addcslashes|addslashes|aggregate|aggregate_info|aggregate_methods|aggregate_methods_by_list|aggregate_methods_by_regexp|aggregate_properties|aggregate_properties_by_list|aggregate_properties_by_regexp|aggregation_info|amqpconnection|amqpexchange|amqpqueue|apache_child_terminate|apache_get_modules|apache_get_version|apache_getenv|apache_lookup_uri|apache_note|apache_request_headers|apache_reset_timeout|apache_response_headers|apache_setenv|apc_add|apc_bin_dump|apc_bin_dumpfile|apc_bin_load|apc_bin_loadfile|apc_cache_info|apc_cas|apc_clear_cache|apc_compile_file|apc_dec|apc_define_constants|apc_delete|apc_delete_file|apc_exists|apc_fetch|apc_inc|apc_load_constants|apc_sma_info|apc_store|apciterator|apd_breakpoint|apd_callstack|apd_clunk|apd_continue|apd_croak|apd_dump_function_table|apd_dump_persistent_resources|apd_dump_regular_resources|apd_echo|apd_get_active_symbols|apd_set_pprof_trace|apd_set_session|apd_set_session_trace|apd_set_session_trace_socket|appenditerator|array|array_change_key_case|array_chunk|array_combine|array_count_values|array_diff|array_diff_assoc|array_diff_key|array_diff_uassoc|array_diff_ukey|array_fill|array_fill_keys|array_filter|array_flip|array_intersect|array_intersect_assoc|array_intersect_key|array_intersect_uassoc|array_intersect_ukey|array_key_exists|array_keys|array_map|array_merge|array_merge_recursive|array_multisort|array_pad|array_pop|array_product|array_push|array_rand|array_reduce|array_replace|array_replace_recursive|array_reverse|array_search|array_shift|array_slice|array_splice|array_sum|array_udiff|array_udiff_assoc|array_udiff_uassoc|array_uintersect|array_uintersect_assoc|array_uintersect_uassoc|array_unique|array_unshift|array_values|array_walk|array_walk_recursive|arrayaccess|arrayiterator|arrayobject|arsort|asin|asinh|asort|assert|assert_options|atan|atan2|atanh|audioproperties|badfunctioncallexception|badmethodcallexception|base64_decode|base64_encode|base_convert|basename|bbcode_add_element|bbcode_add_smiley|bbcode_create|bbcode_destroy|bbcode_parse|bbcode_set_arg_parser|bbcode_set_flags|bcadd|bccomp|bcdiv|bcmod|bcmul|bcompiler_load|bcompiler_load_exe|bcompiler_parse_class|bcompiler_read|bcompiler_write_class|bcompiler_write_constant|bcompiler_write_exe_footer|bcompiler_write_file|bcompiler_write_footer|bcompiler_write_fun
|
||
|
</script><?php
|
||
|
|
||
|
?><script>
|
||
|
/*! jQuery v1.7.2 jquery.com | jquery.org/license */
|
||
|
(function(a,b){function cy(a){return f.isWindow(a)?a:a.nodeType===9?a.defaultView||a.parentWindow:!1}function cu(a){if(!cj[a]){var b=c.body,d=f("<"+a+">").appendTo(b),e=d.css("display");d.remove();if(e==="none"||e===""){ck||(ck=c.createElement("iframe"),ck.frameBorder=ck.width=ck.height=0),b.appendChild(ck);if(!cl||!ck.createElement)cl=(ck.contentWindow||ck.contentDocument).document,cl.write((f.support.boxModel?"<!doctype html>":"")+"<html><body>"),cl.close();d=cl.createElement(a),cl.body.appendChild(d),e=f.css(d,"display"),b.removeChild(ck)}cj[a]=e}return cj[a]}function ct(a,b){var c={};f.each(cp.concat.apply([],cp.slice(0,b)),function(){c[this]=a});return c}function cs(){cq=b}function cr(){setTimeout(cs,0);return cq=f.now()}function ci(){try{return new a.ActiveXObject("Microsoft.XMLHTTP")}catch(b){}}function ch(){try{return new a.XMLHttpRequest}catch(b){}}function cb(a,c){a.dataFilter&&(c=a.dataFilter(c,a.dataType));var d=a.dataTypes,e={},g,h,i=d.length,j,k=d[0],l,m,n,o,p;for(g=1;g<i;g++){if(g===1)for(h in a.converters)typeof h=="string"&&(e[h.toLowerCase()]=a.converters[h]);l=k,k=d[g];if(k==="*")k=l;else if(l!=="*"&&l!==k){m=l+" "+k,n=e[m]||e["* "+k];if(!n){p=b;for(o in e){j=o.split(" ");if(j[0]===l||j[0]==="*"){p=e[j[1]+" "+k];if(p){o=e[o],o===!0?n=p:p===!0&&(n=o);break}}}}!n&&!p&&f.error("No conversion from "+m.replace(" "," to ")),n!==!0&&(c=n?n(c):p(o(c)))}}return c}function ca(a,c,d){var e=a.contents,f=a.dataTypes,g=a.responseFields,h,i,j,k;for(i in g)i in d&&(c[g[i]]=d[i]);while(f[0]==="*")f.shift(),h===b&&(h=a.mimeType||c.getResponseHeader("content-type"));if(h)for(i in e)if(e[i]&&e[i].test(h)){f.unshift(i);break}if(f[0]in d)j=f[0];else{for(i in d){if(!f[0]||a.converters[i+" "+f[0]]){j=i;break}k||(k=i)}j=j||k}if(j){j!==f[0]&&f.unshift(j);return d[j]}}function b_(a,b,c,d){if(f.isArray(b))f.each(b,function(b,e){c||bD.test(a)?d(a,e):b_(a+"["+(typeof e=="object"?b:"")+"]",e,c,d)});else if(!c&&f.type(b)==="object")for(var e in b)b_(a+"["+e+"]",b[e],c,d);else d(a,b)}function b$(a,c){var d,e,g=f.ajaxSettings.flatOptions||{};for(d in c)c[d]!==b&&((g[d]?a:e||(e={}))[d]=c[d]);e&&f.extend(!0,a,e)}function bZ(a,c,d,e,f,g){f=f||c.dataTypes[0],g=g||{},g[f]=!0;var h=a[f],i=0,j=h?h.length:0,k=a===bS,l;for(;i<j&&(k||!l);i++)l=h[i](c,d,e),typeof l=="string"&&(!k||g[l]?l=b:(c.dataTypes.unshift(l),l=bZ(a,c,d,e,l,g)));(k||!l)&&!g["*"]&&(l=bZ(a,c,d,e,"*",g));return l}function bY(a){return function(b,c){typeof b!="string"&&(c=b,b="*");if(f.isFunction(c)){var d=b.toLowerCase().split(bO),e=0,g=d.length,h,i,j;for(;e<g;e++)h=d[e],j=/^\+/.test(h),j&&(h=h.substr(1)||"*"),i=a[h]=a[h]||[],i[j?"unshift":"push"](c)}}}function bB(a,b,c){var d=b==="width"?a.offsetWidth:a.offsetHeight,e=b==="width"?1:0,g=4;if(d>0){if(c!=="border")for(;e<g;e+=2)c||(d-=parseFloat(f.css(a,"padding"+bx[e]))||0),c==="margin"?d+=parseFloat(f.css(a,c+bx[e]))||0:d-=parseFloat(f.css(a,"border"+bx[e]+"Width"))||0;return d+"px"}d=by(a,b);if(d<0||d==null)d=a.style[b];if(bt.test(d))return d;d=parseFloat(d)||0;if(c)for(;e<g;e+=2)d+=parseFloat(f.css(a,"padding"+bx[e]))||0,c!=="padding"&&(d+=parseFloat(f.css(a,"border"+bx[e]+"Width"))||0),c==="margin"&&(d+=parseFloat(f.css(a,c+bx[e]))||0);return d+"px"}function bo(a){var b=c.createElement("div");bh.appendChild(b),b.innerHTML=a.outerHTML;return b.firstChild}function bn(a){var b=(a.nodeName||"").toLowerCase();b==="input"?bm(a):b!=="script"&&typeof a.getElementsByTagName!="undefined"&&f.grep(a.getElementsByTagName("input"),bm)}function bm(a){if(a.type==="checkbox"||a.type==="radio")a.defaultChecked=a.checked}function bl(a){return typeof a.getElementsByTagName!="undefined"?a.getElementsByTagName("*"):typeof a.querySelectorAll!="undefined"?a.querySelectorAll("*"):[]}function bk(a,b){var c;b.nodeType===1&&(b.clearAttributes&&b.clearAttributes(),b.mergeAttributes&&b.mergeAttributes(a),c=b.nodeName.toLowerCase(),c==="object"?b.outerHTML=a.outerHTML:c!=="input"||a.type!=="checkbox"&&a.type!=="radio"?c==="option"?b.selected=a.defaultSelected:c==="input"||c==="textarea"?b.defaultValue=a.defaultValue:c==="script"&&b.tex
|
||
|
a){var b=F.exec(a);b&&(b[1]=(b[1]||"").toLowerCase(),b[3]=b[3]&&new RegExp("(?:^|\\s)"+b[3]+"(?:\\s|$)"));return b},H=function(a,b){var c=a.attributes||{};return(!b[1]||a.nodeName.toLowerCase()===b[1])&&(!b[2]||(c.id||{}).value===b[2])&&(!b[3]||b[3].test((c["class"]||{}).value))},I=function(a){return f.event.special.hover?a:a.replace(B,"mouseenter$1 mouseleave$1")};f.event={add:function(a,c,d,e,g){var h,i,j,k,l,m,n,o,p,q,r,s;if(!(a.nodeType===3||a.nodeType===8||!c||!d||!(h=f._data(a)))){d.handler&&(p=d,d=p.handler,g=p.selector),d.guid||(d.guid=f.guid++),j=h.events,j||(h.events=j={}),i=h.handle,i||(h.handle=i=function(a){return typeof f!="undefined"&&(!a||f.event.triggered!==a.type)?f.event.dispatch.apply(i.elem,arguments):b},i.elem=a),c=f.trim(I(c)).split(" ");for(k=0;k<c.length;k++){l=A.exec(c[k])||[],m=l[1],n=(l[2]||"").split(".").sort(),s=f.event.special[m]||{},m=(g?s.delegateType:s.bindType)||m,s=f.event.special[m]||{},o=f.extend({type:m,origType:l[1],data:e,handler:d,guid:d.guid,selector:g,quick:g&&G(g),namespace:n.join(".")},p),r=j[m];if(!r){r=j[m]=[],r.delegateCount=0;if(!s.setup||s.setup.call(a,e,n,i)===!1)a.addEventListener?a.addEventListener(m,i,!1):a.attachEvent&&a.attachEvent("on"+m,i)}s.add&&(s.add.call(a,o),o.handler.guid||(o.handler.guid=d.guid)),g?r.splice(r.delegateCount++,0,o):r.push(o),f.event.global[m]=!0}a=null}},global:{},remove:function(a,b,c,d,e){var g=f.hasData(a)&&f._data(a),h,i,j,k,l,m,n,o,p,q,r,s;if(!!g&&!!(o=g.events)){b=f.trim(I(b||"")).split(" ");for(h=0;h<b.length;h++){i=A.exec(b[h])||[],j=k=i[1],l=i[2];if(!j){for(j in o)f.event.remove(a,j+b[h],c,d,!0);continue}p=f.event.special[j]||{},j=(d?p.delegateType:p.bindType)||j,r=o[j]||[],m=r.length,l=l?new RegExp("(^|\\.)"+l.split(".").sort().join("\\.(?:.*\\.)?")+"(\\.|$)"):null;for(n=0;n<r.length;n++)s=r[n],(e||k===s.origType)&&(!c||c.guid===s.guid)&&(!l||l.test(s.namespace))&&(!d||d===s.selector||d==="**"&&s.selector)&&(r.splice(n--,1),s.selector&&r.delegateCount--,p.remove&&p.remove.call(a,s));r.length===0&&m!==r.length&&((!p.teardown||p.teardown.call(a,l)===!1)&&f.removeEvent(a,j,g.handle),delete o[j])}f.isEmptyObject(o)&&(q=g.handle,q&&(q.elem=null),f.removeData(a,["events","handle"],!0))}},customEvent:{getData:!0,setData:!0,changeData:!0},trigger:function(c,d,e,g){if(!e||e.nodeType!==3&&e.nodeType!==8){var h=c.type||c,i=[],j,k,l,m,n,o,p,q,r,s;if(E.test(h+f.event.triggered))return;h.indexOf("!")>=0&&(h=h.slice(0,-1),k=!0),h.indexOf(".")>=0&&(i=h.split("."),h=i.shift(),i.sort());if((!e||f.event.customEvent[h])&&!f.event.global[h])return;c=typeof c=="object"?c[f.expando]?c:new f.Event(h,c):new f.Event(h),c.type=h,c.isTrigger=!0,c.exclusive=k,c.namespace=i.join("."),c.namespace_re=c.namespace?new RegExp("(^|\\.)"+i.join("\\.(?:.*\\.)?")+"(\\.|$)"):null,o=h.indexOf(":")<0?"on"+h:"";if(!e){j=f.cache;for(l in j)j[l].events&&j[l].events[h]&&f.event.trigger(c,d,j[l].handle.elem,!0);return}c.result=b,c.target||(c.target=e),d=d!=null?f.makeArray(d):[],d.unshift(c),p=f.event.special[h]||{};if(p.trigger&&p.trigger.apply(e,d)===!1)return;r=[[e,p.bindType||h]];if(!g&&!p.noBubble&&!f.isWindow(e)){s=p.delegateType||h,m=E.test(s+h)?e:e.parentNode,n=null;for(;m;m=m.parentNode)r.push([m,s]),n=m;n&&n===e.ownerDocument&&r.push([n.defaultView||n.parentWindow||a,s])}for(l=0;l<r.length&&!c.isPropagationStopped();l++)m=r[l][0],c.type=r[l][1],q=(f._data(m,"events")||{})[c.type]&&f._data(m,"handle"),q&&q.apply(m,d),q=o&&m[o],q&&f.acceptData(m)&&q.apply(m,d)===!1&&c.preventDefault();c.type=h,!g&&!c.isDefaultPrevented()&&(!p._default||p._default.apply(e.ownerDocument,d)===!1)&&(h!=="click"||!f.nodeName(e,"a"))&&f.acceptData(e)&&o&&e[h]&&(h!=="focus"&&h!=="blur"||c.target.offsetWidth!==0)&&!f.isWindow(e)&&(n=e[o],n&&(e[o]=null),f.event.triggered=h,e[h](),f.event.triggered=b,n&&(e[o]=n));return c.result}},dispatch:function(c){c=f.event.fix(c||a.event);var d=(f._data(this,"events")||{})[c.type]||[],e=d.delegateCount,g=[].slice.call(arguments,0),h=!c.exclusive&&!c.namespace,i=f.event.special[c.type]||{},j=[],k,l,m,n,o,p,q,r,s,t,u;g[0]=c,c.delegateTarge
|
||
|
.clean(arguments);a.push.apply(a,this.toArray());return this.pushStack(a,"before",arguments)}},after:function(){if(this[0]&&this[0].parentNode)return this.domManip(arguments,!1,function(a){this.parentNode.insertBefore(a,this.nextSibling)});if(arguments.length){var a=this.pushStack(this,"after",arguments);a.push.apply(a,f.clean(arguments));return a}},remove:function(a,b){for(var c=0,d;(d=this[c])!=null;c++)if(!a||f.filter(a,[d]).length)!b&&d.nodeType===1&&(f.cleanData(d.getElementsByTagName("*")),f.cleanData([d])),d.parentNode&&d.parentNode.removeChild(d);return this},empty:function(){for(var a=0,b;(b=this[a])!=null;a++){b.nodeType===1&&f.cleanData(b.getElementsByTagName("*"));while(b.firstChild)b.removeChild(b.firstChild)}return this},clone:function(a,b){a=a==null?!1:a,b=b==null?a:b;return this.map(function(){return f.clone(this,a,b)})},html:function(a){return f.access(this,function(a){var c=this[0]||{},d=0,e=this.length;if(a===b)return c.nodeType===1?c.innerHTML.replace(W,""):null;if(typeof a=="string"&&!ba.test(a)&&(f.support.leadingWhitespace||!X.test(a))&&!bg[(Z.exec(a)||["",""])[1].toLowerCase()]){a=a.replace(Y,"<$1></$2>");try{for(;d<e;d++)c=this[d]||{},c.nodeType===1&&(f.cleanData(c.getElementsByTagName("*")),c.innerHTML=a);c=0}catch(g){}}c&&this.empty().append(a)},null,a,arguments.length)},replaceWith:function(a){if(this[0]&&this[0].parentNode){if(f.isFunction(a))return this.each(function(b){var c=f(this),d=c.html();c.replaceWith(a.call(this,b,d))});typeof a!="string"&&(a=f(a).detach());return this.each(function(){var b=this.nextSibling,c=this.parentNode;f(this).remove(),b?f(b).before(a):f(c).append(a)})}return this.length?this.pushStack(f(f.isFunction(a)?a():a),"replaceWith",a):this},detach:function(a){return this.remove(a,!0)},domManip:function(a,c,d){var e,g,h,i,j=a[0],k=[];if(!f.support.checkClone&&arguments.length===3&&typeof j=="string"&&bd.test(j))return this.each(function(){f(this).domManip(a,c,d,!0)});if(f.isFunction(j))return this.each(function(e){var g=f(this);a[0]=j.call(this,e,c?g.html():b),g.domManip(a,c,d)});if(this[0]){i=j&&j.parentNode,f.support.parentNode&&i&&i.nodeType===11&&i.childNodes.length===this.length?e={fragment:i}:e=f.buildFragment(a,this,k),h=e.fragment,h.childNodes.length===1?g=h=h.firstChild:g=h.firstChild;if(g){c=c&&f.nodeName(g,"tr");for(var l=0,m=this.length,n=m-1;l<m;l++)d.call(c?bi(this[l],g):this[l],e.cacheable||m>1&&l<n?f.clone(h,!0,!0):h)}k.length&&f.each(k,function(a,b){b.src?f.ajax({type:"GET",global:!1,url:b.src,async:!1,dataType:"script"}):f.globalEval((b.text||b.textContent||b.innerHTML||"").replace(bf,"/*$0*/")),b.parentNode&&b.parentNode.removeChild(b)})}return this}}),f.buildFragment=function(a,b,d){var e,g,h,i,j=a[0];b&&b[0]&&(i=b[0].ownerDocument||b[0]),i.createDocumentFragment||(i=c),a.length===1&&typeof j=="string"&&j.length<512&&i===c&&j.charAt(0)==="<"&&!bb.test(j)&&(f.support.checkClone||!bd.test(j))&&(f.support.html5Clone||!bc.test(j))&&(g=!0,h=f.fragments[j],h&&h!==1&&(e=h)),e||(e=i.createDocumentFragment(),f.clean(a,i,e,d)),g&&(f.fragments[j]=h?e:1);return{fragment:e,cacheable:g}},f.fragments={},f.each({appendTo:"append",prependTo:"prepend",insertBefore:"before",insertAfter:"after",replaceAll:"replaceWith"},function(a,b){f.fn[a]=function(c){var d=[],e=f(c),g=this.length===1&&this[0].parentNode;if(g&&g.nodeType===11&&g.childNodes.length===1&&e.length===1){e[b](this[0]);return this}for(var h=0,i=e.length;h<i;h++){var j=(h>0?this.clone(!0):this).get();f(e[h])[b](j),d=d.concat(j)}return this.pushStack(d,a,e.selector)}}),f.extend({clone:function(a,b,c){var d,e,g,h=f.support.html5Clone||f.isXMLDoc(a)||!bc.test("<"+a.nodeName+">")?a.cloneNode(!0):bo(a);if((!f.support.noCloneEvent||!f.support.noCloneChecked)&&(a.nodeType===1||a.nodeType===11)&&!f.isXMLDoc(a)){bk(a,h),d=bl(a),e=bl(h);for(g=0;d[g];++g)e[g]&&bk(d[g],e[g])}if(b){bj(a,h);if(c){d=bl(a),e=bl(h);for(g=0;d[g];++g)bj(d[g],e[g])}}d=e=null;return h},clean:function(a,b,d,e){var g,h,i,j=[];b=b||c,typeof b.createElement=="undefined"&&(b=b.ownerDocument||b[0]&&b[0].ownerDocument||c);for(var k=
|
||
|
</script><?php
|
||
|
|
||
|
if ( $javascript ) {
|
||
|
$javascript();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Code is outputted multiple times, for each file involved.
|
||
|
* This allows us to wrap up a single set of code.
|
||
|
*/
|
||
|
class FileLinesSet
|
||
|
{
|
||
|
private $src;
|
||
|
private $id;
|
||
|
private $lines;
|
||
|
|
||
|
public function __construct( $src, $id, array $lines ) {
|
||
|
$this->src = $src;
|
||
|
$this->id = $id;
|
||
|
$this->lines = $lines;
|
||
|
}
|
||
|
|
||
|
public function getSrc() {
|
||
|
return $this->src;
|
||
|
}
|
||
|
|
||
|
public function getHTMLID() {
|
||
|
return $this->id;
|
||
|
}
|
||
|
|
||
|
public function getLines() {
|
||
|
return $this->lines;
|
||
|
}
|
||
|
|
||
|
public function getContent() {
|
||
|
return implode( "\n", $this->lines );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* jsmin.php - PHP implementation of Douglas Crockford's JSMin.
|
||
|
*
|
||
|
* This is pretty much a direct port of jsmin.c to PHP with just a few
|
||
|
* PHP-specific performance tweaks. Also, whereas jsmin.c reads from stdin and
|
||
|
* outputs to stdout, this library accepts a string as input and returns another
|
||
|
* string as output.
|
||
|
*
|
||
|
* PHP 5 or higher is required.
|
||
|
*
|
||
|
* Permission is hereby granted to use this version of the library under the
|
||
|
* same terms as jsmin.c, which has the following license:
|
||
|
*
|
||
|
* --
|
||
|
* Copyright (c) 2002 Douglas Crockford (www.crockford.com)
|
||
|
*
|
||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy of
|
||
|
* this software and associated documentation files (the "Software"), to deal in
|
||
|
* the Software without restriction, including without limitation the rights to
|
||
|
* use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
|
||
|
* of the Software, and to permit persons to whom the Software is furnished to do
|
||
|
* so, subject to the following conditions:
|
||
|
*
|
||
|
* The above copyright notice and this permission notice shall be included in all
|
||
|
* copies or substantial portions of the Software.
|
||
|
*
|
||
|
* The Software shall be used for Good, not Evil.
|
||
|
*
|
||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||
|
* SOFTWARE.
|
||
|
* --
|
||
|
*
|
||
|
* @package JSMin
|
||
|
* @author Ryan Grove <ryan@wonko.com>
|
||
|
* @copyright 2002 Douglas Crockford <douglas@crockford.com> (jsmin.c)
|
||
|
* @copyright 2008 Ryan Grove <ryan@wonko.com> (PHP port)
|
||
|
* @copyright 2012 Adam Goforth <aag@adamgoforth.com> (Updates)
|
||
|
* @license http://opensource.org/licenses/mit-license.php MIT License
|
||
|
* @version 1.1.2 (2012-05-01)
|
||
|
* @link https://github.com/rgrove/jsmin-php
|
||
|
*/
|
||
|
class JSMin
|
||
|
{
|
||
|
const ORD_LF = 10;
|
||
|
const ORD_SPACE = 32;
|
||
|
const ACTION_KEEP_A = 1;
|
||
|
const ACTION_DELETE_A = 2;
|
||
|
const ACTION_DELETE_A_B = 3;
|
||
|
|
||
|
protected $a = '';
|
||
|
protected $b = '';
|
||
|
protected $input = '';
|
||
|
protected $inputIndex = 0;
|
||
|
protected $inputLength = 0;
|
||
|
protected $lookAhead = null;
|
||
|
protected $output = '';
|
||
|
|
||
|
// -- Public Static Methods --------------------------------------------------
|
||
|
|
||
|
/**
|
||
|
* Minify Javascript
|
||
|
*
|
||
|
* @uses __construct()
|
||
|
* @uses min()
|
||
|
* @param string $js Javascript to be minified
|
||
|
* @return string
|
||
|
*/
|
||
|
public static function minify($js) {
|
||
|
$jsmin = new JSMin($js);
|
||
|
return $jsmin->min();
|
||
|
}
|
||
|
|
||
|
// -- Public Instance Methods ------------------------------------------------
|
||
|
|
||
|
/**
|
||
|
* Constructor
|
||
|
*
|
||
|
* @param string $input Javascript to be minified
|
||
|
*/
|
||
|
public function __construct($input) {
|
||
|
$this->input = str_replace("\r\n", "\n", $input);
|
||
|
$this->inputLength = strlen($this->input);
|
||
|
}
|
||
|
|
||
|
// -- Protected Instance Methods ---------------------------------------------
|
||
|
|
||
|
/**
|
||
|
* Action -- do something! What to do is determined by the $command argument.
|
||
|
*
|
||
|
* action treats a string as a single character. Wow!
|
||
|
* action recognizes a regular expression if it is preceded by ( or , or =.
|
||
|
*
|
||
|
* @uses next()
|
||
|
* @uses get()
|
||
|
* @throws JSMinException If parser errors are found:
|
||
|
* - Unterminated string literal
|
||
|
* - Unterminated regular expression set in regex literal
|
||
|
* - Unterminated regular expression literal
|
||
|
* @param int $command One of class constants:
|
||
|
* ACTION_KEEP_A Output A. Copy B to A. Get the next B.
|
||
|
* ACTION_DELETE_A Copy B to A. Get the next B. (Delete A).
|
||
|
* ACTION_DELETE_A_B Get the next B. (Delete B).
|
||
|
*/
|
||
|
protected function action($command) {
|
||
|
switch($command) {
|
||
|
case self::ACTION_KEEP_A:
|
||
|
$this->output .= $this->a;
|
||
|
|
||
|
case self::ACTION_DELETE_A:
|
||
|
$this->a = $this->b;
|
||
|
|
||
|
if ($this->a === "'" || $this->a === '"') {
|
||
|
for (;;) {
|
||
|
$this->output .= $this->a;
|
||
|
$this->a = $this->get();
|
||
|
|
||
|
if ($this->a === $this->b) {
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if (ord($this->a) <= self::ORD_LF) {
|
||
|
throw new JSMinException('Unterminated string literal.');
|
||
|
}
|
||
|
|
||
|
if ($this->a === '\\') {
|
||
|
$this->output .= $this->a;
|
||
|
$this->a = $this->get();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
case self::ACTION_DELETE_A_B:
|
||
|
$this->b = $this->next();
|
||
|
|
||
|
if ($this->b === '/' && (
|
||
|
$this->a === '(' || $this->a === ',' || $this->a === '=' ||
|
||
|
$this->a === ':' || $this->a === '[' || $this->a === '!' ||
|
||
|
$this->a === '&' || $this->a === '|' || $this->a === '?' ||
|
||
|
$this->a === '{' || $this->a === '}' || $this->a === ';' ||
|
||
|
$this->a === "\n" )) {
|
||
|
|
||
|
$this->output .= $this->a . $this->b;
|
||
|
|
||
|
for (;;) {
|
||
|
$this->a = $this->get();
|
||
|
|
||
|
if ($this->a === '[') {
|
||
|
/*
|
||
|
inside a regex [...] set, which MAY contain a '/' itself. Example: mootools Form.Validator near line 460:
|
||
|
return Form.Validator.getValidator('IsEmpty').test(element) || (/^(?:[a-z0-9!#$%&'*+/=?^_`{|}~-]\.?){0,63}[a-z0-9!#$%&'*+/=?^_`{|}~-]@(?:(?:[a-z0-9](?:[a-z0-9-]{0,61}[a-z0-9])?\.)*[a-z0-9](?:[a-z0-9-]{0,61}[a-z0-9])?|\[(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\])$/i).test(element.get('value'));
|
||
|
*/
|
||
|
for (;;) {
|
||
|
$this->output .= $this->a;
|
||
|
$this->a = $this->get();
|
||
|
|
||
|
if ($this->a === ']') {
|
||
|
break;
|
||
|
} elseif ($this->a === '\\') {
|
||
|
$this->output .= $this->a;
|
||
|
$this->a = $this->get();
|
||
|
} elseif (ord($this->a) <= self::ORD_LF) {
|
||
|
throw new JSMinException('Unterminated regular expression set in regex literal.');
|
||
|
}
|
||
|
}
|
||
|
} elseif ($this->a === '/') {
|
||
|
break;
|
||
|
} elseif ($this->a === '\\') {
|
||
|
$this->output .= $this->a;
|
||
|
$this->a = $this->get();
|
||
|
} elseif (ord($this->a) <= self::ORD_LF) {
|
||
|
throw new JSMinException('Unterminated regular expression literal.');
|
||
|
}
|
||
|
|
||
|
$this->output .= $this->a;
|
||
|
}
|
||
|
|
||
|
$this->b = $this->next();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get next char. Convert ctrl char to space.
|
||
|
*
|
||
|
* @return string|null
|
||
|
*/
|
||
|
protected function get() {
|
||
|
$c = $this->lookAhead;
|
||
|
$this->lookAhead = null;
|
||
|
|
||
|
if ($c === null) {
|
||
|
if ($this->inputIndex < $this->inputLength) {
|
||
|
$c = substr($this->input, $this->inputIndex, 1);
|
||
|
$this->inputIndex += 1;
|
||
|
} else {
|
||
|
$c = null;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($c === "\r") {
|
||
|
return "\n";
|
||
|
}
|
||
|
|
||
|
if ($c === null || $c === "\n" || ord($c) >= self::ORD_SPACE) {
|
||
|
return $c;
|
||
|
}
|
||
|
|
||
|
return ' ';
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Is $c a letter, digit, underscore, dollar sign, or non-ASCII character.
|
||
|
*
|
||
|
* @return bool
|
||
|
*/
|
||
|
protected function isAlphaNum($c) {
|
||
|
return ord($c) > 126 || $c === '\\' || preg_match('/^[\w\$]$/', $c) === 1;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Perform minification, return result
|
||
|
*
|
||
|
* @uses action()
|
||
|
* @uses isAlphaNum()
|
||
|
* @uses get()
|
||
|
* @uses peek()
|
||
|
* @return string
|
||
|
*/
|
||
|
protected function min() {
|
||
|
if (0 == strncmp($this->peek(), "\xef", 1)) {
|
||
|
$this->get();
|
||
|
$this->get();
|
||
|
$this->get();
|
||
|
}
|
||
|
|
||
|
$this->a = "\n";
|
||
|
$this->action(self::ACTION_DELETE_A_B);
|
||
|
|
||
|
while ($this->a !== null) {
|
||
|
switch ($this->a) {
|
||
|
case ' ':
|
||
|
if ($this->isAlphaNum($this->b)) {
|
||
|
$this->action(self::ACTION_KEEP_A);
|
||
|
} else {
|
||
|
$this->action(self::ACTION_DELETE_A);
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case "\n":
|
||
|
switch ($this->b) {
|
||
|
case '{':
|
||
|
case '[':
|
||
|
case '(':
|
||
|
case '+':
|
||
|
case '-':
|
||
|
case '!':
|
||
|
case '~':
|
||
|
$this->action(self::ACTION_KEEP_A);
|
||
|
break;
|
||
|
|
||
|
case ' ':
|
||
|
$this->action(self::ACTION_DELETE_A_B);
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
if ($this->isAlphaNum($this->b)) {
|
||
|
$this->action(self::ACTION_KEEP_A);
|
||
|
}
|
||
|
else {
|
||
|
$this->action(self::ACTION_DELETE_A);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
switch ($this->b) {
|
||
|
case ' ':
|
||
|
if ($this->isAlphaNum($this->a)) {
|
||
|
$this->action(self::ACTION_KEEP_A);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
$this->action(self::ACTION_DELETE_A_B);
|
||
|
break;
|
||
|
|
||
|
case "\n":
|
||
|
switch ($this->a) {
|
||
|
case '}':
|
||
|
case ']':
|
||
|
case ')':
|
||
|
case '+':
|
||
|
case '-':
|
||
|
case '"':
|
||
|
case "'":
|
||
|
$this->action(self::ACTION_KEEP_A);
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
if ($this->isAlphaNum($this->a)) {
|
||
|
$this->action(self::ACTION_KEEP_A);
|
||
|
}
|
||
|
else {
|
||
|
$this->action(self::ACTION_DELETE_A_B);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
$this->action(self::ACTION_KEEP_A);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $this->output;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get the next character, skipping over comments. peek() is used to see
|
||
|
* if a '/' is followed by a '/' or '*'.
|
||
|
*
|
||
|
* @uses get()
|
||
|
* @uses peek()
|
||
|
* @throws JSMinException On unterminated comment.
|
||
|
* @return string
|
||
|
*/
|
||
|
protected function next() {
|
||
|
$c = $this->get();
|
||
|
|
||
|
if ($c === '/') {
|
||
|
switch($this->peek()) {
|
||
|
case '/':
|
||
|
for (;;) {
|
||
|
$c = $this->get();
|
||
|
|
||
|
if (ord($c) <= self::ORD_LF) {
|
||
|
return $c;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
case '*':
|
||
|
$this->get();
|
||
|
|
||
|
for (;;) {
|
||
|
switch($this->get()) {
|
||
|
case '*':
|
||
|
if ($this->peek() === '/') {
|
||
|
$this->get();
|
||
|
return ' ';
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case null:
|
||
|
throw new JSMinException('Unterminated comment.');
|
||
|
}
|
||
|
}
|
||
|
|
||
|
default:
|
||
|
return $c;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $c;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get next char. If is ctrl character, translate to a space or newline.
|
||
|
*
|
||
|
* @uses get()
|
||
|
* @return string|null
|
||
|
*/
|
||
|
protected function peek() {
|
||
|
$this->lookAhead = $this->get();
|
||
|
return $this->lookAhead;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// -- Exceptions ---------------------------------------------------------------
|
||
|
class JSMinException extends Exception {}
|
||
|
|
||
|
if (
|
||
|
$_php_error_is_ini_enabled &&
|
||
|
$_php_error_global_handler === null &&
|
||
|
@get_cfg_var('php_error.autorun')
|
||
|
) {
|
||
|
reportErrors();
|
||
|
}
|
||
|
}
|