A super easy PHP Framework for web development!
https://github.com/exacti/phacil-framework
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
446 lines
11 KiB
446 lines
11 KiB
10 months ago
|
<?php
|
||
|
/**
|
||
|
* Copyright © 2024 ExacTI Technology Solutions. All rights reserved.
|
||
|
* GPLv3 General License.
|
||
|
* https://exacti.com.br
|
||
|
* Phacil PHP Framework - https://github.com/exacti/phacil-framework
|
||
|
*/
|
||
|
|
||
|
namespace Phacil\Framework\ECompress\Helpers\JSMinPlus;
|
||
|
|
||
|
use Phacil\Framework\Exception;
|
||
|
use Phacil\Framework\ECompress\Helpers\JSMinPlus\JSToken;
|
||
|
|
||
|
/**
|
||
|
* JSMinPlus version 1.4
|
||
|
*
|
||
|
* Minifies a javascript file using a javascript parser
|
||
|
*
|
||
|
* This implements a PHP port of Brendan Eich's Narcissus open source javascript engine (in javascript)
|
||
|
* References: http://en.wikipedia.org/wiki/Narcissus_(JavaScript_engine)
|
||
|
* Narcissus sourcecode: http://mxr.mozilla.org/mozilla/source/js/narcissus/
|
||
|
* JSMinPlus weblog: http://crisp.tweakblogs.net/blog/cat/716
|
||
|
*
|
||
|
* Tino Zijdel <crisp@tweakers.net>
|
||
|
*
|
||
|
* Usage: $minified = JSMinPlus::minify($script [, $filename])
|
||
|
*
|
||
|
* Versionlog (see also changelog.txt):
|
||
|
* 23-07-2011 - remove dynamic creation of OP_* and KEYWORD_* defines and declare them on top
|
||
|
* reduce memory footprint by minifying by block-scope
|
||
|
* some small byte-saving and performance improvements
|
||
|
* 12-05-2009 - fixed hook:colon precedence, fixed empty body in loop and if-constructs
|
||
|
* 18-04-2009 - fixed crashbug in PHP 5.2.9 and several other bugfixes
|
||
|
* 12-04-2009 - some small bugfixes and performance improvements
|
||
|
* 09-04-2009 - initial open sourced version 1.0
|
||
|
*
|
||
|
* Latest version of this script: http://files.tweakers.net/jsminplus/jsminplus.zip
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
/* ***** BEGIN LICENSE BLOCK *****
|
||
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
||
|
*
|
||
|
* The contents of this file are subject to the Mozilla Public License Version
|
||
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
||
|
* the License. You may obtain a copy of the License at
|
||
|
* http://www.mozilla.org/MPL/
|
||
|
*
|
||
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
||
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
||
|
* for the specific language governing rights and limitations under the
|
||
|
* License.
|
||
|
*
|
||
|
* The Original Code is the Narcissus JavaScript engine.
|
||
|
*
|
||
|
* The Initial Developer of the Original Code is
|
||
|
* Brendan Eich <brendan@mozilla.org>.
|
||
|
* Portions created by the Initial Developer are Copyright (C) 2004
|
||
|
* the Initial Developer. All Rights Reserved.
|
||
|
*
|
||
|
* Contributor(s): Tino Zijdel <crisp@tweakers.net>
|
||
|
* PHP port, modifications and minifier routine are (C) 2009-2011
|
||
|
*
|
||
|
* Alternatively, the contents of this file may be used under the terms of
|
||
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
||
|
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
||
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
||
|
* of those above. If you wish to allow use of your version of this file only
|
||
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
||
|
* use your version of this file under the terms of the MPL, indicate your
|
||
|
* decision by deleting the provisions above and replace them with the notice
|
||
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
||
|
* the provisions above, a recipient may use your version of this file under
|
||
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
||
|
*
|
||
|
* ***** END LICENSE BLOCK ***** */
|
||
|
|
||
|
class JSTokenizer
|
||
|
{
|
||
|
private $cursor = 0;
|
||
|
private $source;
|
||
|
|
||
|
public $tokens = array();
|
||
|
public $tokenIndex = 0;
|
||
|
public $lookahead = 0;
|
||
|
public $scanNewlines = false;
|
||
|
public $scanOperand = true;
|
||
|
|
||
|
public $filename;
|
||
|
public $lineno;
|
||
|
|
||
|
private $keywords = array(
|
||
|
'break',
|
||
|
'case', 'catch', 'const', 'continue',
|
||
|
'debugger', 'default', 'delete', 'do',
|
||
|
'else', 'enum',
|
||
|
'false', 'finally', 'for', 'function',
|
||
|
'if', 'in', 'instanceof',
|
||
|
'new', 'null',
|
||
|
'return',
|
||
|
'switch',
|
||
|
'this', 'throw', 'true', 'try', 'typeof',
|
||
|
'var', 'void',
|
||
|
'while', 'with'
|
||
|
);
|
||
|
|
||
|
private $opTypeNames = array(
|
||
|
';', ',', '?', ':', '||', '&&', '|', '^',
|
||
|
'&', '===', '==', '=', '!==', '!=', '<<', '<=',
|
||
|
'<', '>>>', '>>', '>=', '>', '++', '--', '+',
|
||
|
'-', '*', '/', '%', '!', '~', '.', '[',
|
||
|
']', '{', '}', '(', ')', '@*/'
|
||
|
);
|
||
|
|
||
|
private $assignOps = array('|', '^', '&', '<<', '>>', '>>>', '+', '-', '*', '/', '%');
|
||
|
private $opRegExp;
|
||
|
|
||
|
public function __construct()
|
||
|
{
|
||
|
$this->opRegExp = '#^(' . implode('|', array_map('preg_quote', $this->opTypeNames)) . ')#';
|
||
|
}
|
||
|
|
||
|
public function init($source, $filename = '', $lineno = 1)
|
||
|
{
|
||
|
$this->source = $source;
|
||
|
$this->filename = $filename ? $filename : '[inline]';
|
||
|
$this->lineno = $lineno;
|
||
|
|
||
|
$this->cursor = 0;
|
||
|
$this->tokens = array();
|
||
|
$this->tokenIndex = 0;
|
||
|
$this->lookahead = 0;
|
||
|
$this->scanNewlines = false;
|
||
|
$this->scanOperand = true;
|
||
|
}
|
||
|
|
||
|
public function getInput($chunksize)
|
||
|
{
|
||
|
if ($chunksize)
|
||
|
return substr($this->source, $this->cursor, $chunksize);
|
||
|
|
||
|
return substr($this->source, $this->cursor);
|
||
|
}
|
||
|
|
||
|
public function isDone()
|
||
|
{
|
||
|
return $this->peek() == TOKEN_END;
|
||
|
}
|
||
|
|
||
|
public function match($tt)
|
||
|
{
|
||
|
return $this->get() == $tt || $this->unget();
|
||
|
}
|
||
|
|
||
|
public function mustMatch($tt)
|
||
|
{
|
||
|
if (!$this->match($tt))
|
||
|
throw $this->newSyntaxError('Unexpected token; token ' . $tt . ' expected');
|
||
|
|
||
|
return $this->currentToken();
|
||
|
}
|
||
|
|
||
|
public function peek()
|
||
|
{
|
||
|
if ($this->lookahead)
|
||
|
{
|
||
|
$next = $this->tokens[($this->tokenIndex + $this->lookahead) & 3];
|
||
|
if ($this->scanNewlines && $next->lineno != $this->lineno)
|
||
|
$tt = TOKEN_NEWLINE;
|
||
|
else
|
||
|
$tt = $next->type;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
$tt = $this->get();
|
||
|
$this->unget();
|
||
|
}
|
||
|
|
||
|
return $tt;
|
||
|
}
|
||
|
|
||
|
public function peekOnSameLine()
|
||
|
{
|
||
|
$this->scanNewlines = true;
|
||
|
$tt = $this->peek();
|
||
|
$this->scanNewlines = false;
|
||
|
|
||
|
return $tt;
|
||
|
}
|
||
|
|
||
|
public function currentToken()
|
||
|
{
|
||
|
if (!empty($this->tokens))
|
||
|
return $this->tokens[$this->tokenIndex];
|
||
|
}
|
||
|
|
||
|
public function get($chunksize = 1000)
|
||
|
{
|
||
|
while($this->lookahead)
|
||
|
{
|
||
|
$this->lookahead--;
|
||
|
$this->tokenIndex = ($this->tokenIndex + 1) & 3;
|
||
|
$token = $this->tokens[$this->tokenIndex];
|
||
|
if ($token->type != TOKEN_NEWLINE || $this->scanNewlines)
|
||
|
return $token->type;
|
||
|
}
|
||
|
|
||
|
$conditional_comment = false;
|
||
|
|
||
|
// strip whitespace and comments
|
||
|
while(true)
|
||
|
{
|
||
|
$input = $this->getInput($chunksize);
|
||
|
|
||
|
// whitespace handling; gobble up \r as well (effectively we don't have support for MAC newlines!)
|
||
|
$re = $this->scanNewlines ? '/^[ \r\t]+/' : '/^\s+/';
|
||
|
if (preg_match($re, $input, $match))
|
||
|
{
|
||
|
$spaces = $match[0];
|
||
|
$spacelen = strlen($spaces);
|
||
|
$this->cursor += $spacelen;
|
||
|
if (!$this->scanNewlines)
|
||
|
$this->lineno += substr_count($spaces, "\n");
|
||
|
|
||
|
if ($spacelen == $chunksize)
|
||
|
continue; // complete chunk contained whitespace
|
||
|
|
||
|
$input = $this->getInput($chunksize);
|
||
|
if ($input == '' || $input[0] != '/')
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
// Comments
|
||
|
if (!preg_match('/^\/(?:\*(@(?:cc_on|if|elif|else|end))?.*?\*\/|\/[^\n]*)/s', $input, $match))
|
||
|
{
|
||
|
if (!$chunksize)
|
||
|
break;
|
||
|
|
||
|
// retry with a full chunk fetch; this also prevents breakage of long regular expressions (which will never match a comment)
|
||
|
$chunksize = null;
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// check if this is a conditional (JScript) comment
|
||
|
if (!empty($match[1]))
|
||
|
{
|
||
|
$match[0] = '/*' . $match[1];
|
||
|
$conditional_comment = true;
|
||
|
break;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
$this->cursor += strlen($match[0]);
|
||
|
$this->lineno += substr_count($match[0], "\n");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($input == '')
|
||
|
{
|
||
|
$tt = TOKEN_END;
|
||
|
$match = array('');
|
||
|
}
|
||
|
elseif ($conditional_comment)
|
||
|
{
|
||
|
$tt = TOKEN_CONDCOMMENT_START;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
switch ($input[0])
|
||
|
{
|
||
|
case '0':
|
||
|
// hexadecimal
|
||
|
if (($input[1] == 'x' || $input[1] == 'X') && preg_match('/^0x[0-9a-f]+/i', $input, $match))
|
||
|
{
|
||
|
$tt = TOKEN_NUMBER;
|
||
|
break;
|
||
|
}
|
||
|
// FALL THROUGH
|
||
|
|
||
|
case '1': case '2': case '3': case '4': case '5':
|
||
|
case '6': case '7': case '8': case '9':
|
||
|
// should always match
|
||
|
preg_match('/^\d+(?:\.\d*)?(?:[eE][-+]?\d+)?/', $input, $match);
|
||
|
$tt = TOKEN_NUMBER;
|
||
|
break;
|
||
|
|
||
|
case "'":
|
||
|
if (preg_match('/^\'(?:[^\\\\\'\r\n]++|\\\\(?:.|\r?\n))*\'/', $input, $match))
|
||
|
{
|
||
|
$tt = TOKEN_STRING;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if ($chunksize)
|
||
|
return $this->get(null); // retry with a full chunk fetch
|
||
|
|
||
|
throw $this->newSyntaxError('Unterminated string literal');
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case '"':
|
||
|
if (preg_match('/^"(?:[^\\\\"\r\n]++|\\\\(?:.|\r?\n))*"/', $input, $match))
|
||
|
{
|
||
|
$tt = TOKEN_STRING;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if ($chunksize)
|
||
|
return $this->get(null); // retry with a full chunk fetch
|
||
|
|
||
|
throw $this->newSyntaxError('Unterminated string literal');
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case '/':
|
||
|
if ($this->scanOperand && preg_match('/^\/((?:\\\\.|\[(?:\\\\.|[^\]])*\]|[^\/])+)\/([gimy]*)/', $input, $match))
|
||
|
{
|
||
|
$tt = TOKEN_REGEXP;
|
||
|
break;
|
||
|
}
|
||
|
// FALL THROUGH
|
||
|
|
||
|
case '|':
|
||
|
case '^':
|
||
|
case '&':
|
||
|
case '<':
|
||
|
case '>':
|
||
|
case '+':
|
||
|
case '-':
|
||
|
case '*':
|
||
|
case '%':
|
||
|
case '=':
|
||
|
case '!':
|
||
|
// should always match
|
||
|
preg_match($this->opRegExp, $input, $match);
|
||
|
$op = $match[0];
|
||
|
if (in_array($op, $this->assignOps) && $input[strlen($op)] == '=')
|
||
|
{
|
||
|
$tt = OP_ASSIGN;
|
||
|
$match[0] .= '=';
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
$tt = $op;
|
||
|
if ($this->scanOperand)
|
||
|
{
|
||
|
if ($op == OP_PLUS)
|
||
|
$tt = OP_UNARY_PLUS;
|
||
|
elseif ($op == OP_MINUS)
|
||
|
$tt = OP_UNARY_MINUS;
|
||
|
}
|
||
|
$op = null;
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case '.':
|
||
|
if (preg_match('/^\.\d+(?:[eE][-+]?\d+)?/', $input, $match))
|
||
|
{
|
||
|
$tt = TOKEN_NUMBER;
|
||
|
break;
|
||
|
}
|
||
|
// FALL THROUGH
|
||
|
|
||
|
case ';':
|
||
|
case ',':
|
||
|
case '?':
|
||
|
case ':':
|
||
|
case '~':
|
||
|
case '[':
|
||
|
case ']':
|
||
|
case '{':
|
||
|
case '}':
|
||
|
case '(':
|
||
|
case ')':
|
||
|
// these are all single
|
||
|
$match = array($input[0]);
|
||
|
$tt = $input[0];
|
||
|
break;
|
||
|
|
||
|
case '@':
|
||
|
// check end of conditional comment
|
||
|
if (substr($input, 0, 3) == '@*/')
|
||
|
{
|
||
|
$match = array('@*/');
|
||
|
$tt = TOKEN_CONDCOMMENT_END;
|
||
|
}
|
||
|
else
|
||
|
throw $this->newSyntaxError('Illegal token');
|
||
|
break;
|
||
|
|
||
|
case "\n":
|
||
|
if ($this->scanNewlines)
|
||
|
{
|
||
|
$match = array("\n");
|
||
|
$tt = TOKEN_NEWLINE;
|
||
|
}
|
||
|
else
|
||
|
throw $this->newSyntaxError('Illegal token');
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
// FIXME: add support for unicode and unicode escape sequence \uHHHH
|
||
|
if (preg_match('/^[$\w]+/', $input, $match))
|
||
|
{
|
||
|
$tt = in_array($match[0], $this->keywords) ? $match[0] : TOKEN_IDENTIFIER;
|
||
|
}
|
||
|
else
|
||
|
throw $this->newSyntaxError('Illegal token');
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$this->tokenIndex = ($this->tokenIndex + 1) & 3;
|
||
|
|
||
|
if (!isset($this->tokens[$this->tokenIndex]))
|
||
|
$this->tokens[$this->tokenIndex] = new JSToken();
|
||
|
|
||
|
$token = $this->tokens[$this->tokenIndex];
|
||
|
$token->type = $tt;
|
||
|
|
||
|
if ($tt == OP_ASSIGN)
|
||
|
$token->assignOp = $op;
|
||
|
|
||
|
$token->start = $this->cursor;
|
||
|
|
||
|
$token->value = $match[0];
|
||
|
$this->cursor += strlen($match[0]);
|
||
|
|
||
|
$token->end = $this->cursor;
|
||
|
$token->lineno = $this->lineno;
|
||
|
|
||
|
return $tt;
|
||
|
}
|
||
|
|
||
|
public function unget()
|
||
|
{
|
||
|
if (++$this->lookahead == 4)
|
||
|
throw $this->newSyntaxError('PANIC: too much lookahead!');
|
||
|
|
||
|
$this->tokenIndex = ($this->tokenIndex - 1) & 3;
|
||
|
}
|
||
|
|
||
|
public function newSyntaxError($m)
|
||
|
{
|
||
|
return new Exception('Parse error: ' . $m . ' in file \'' . $this->filename . '\' on line ' . $this->lineno);
|
||
|
}
|
||
|
}
|