184 lines
4.5 KiB
PHP
184 lines
4.5 KiB
PHP
|
<?php
|
||
|
/*+*******************************************************************************
|
||
|
* The contents of this file are subject to the vtiger CRM Public License Version 1.0
|
||
|
* ("License"); You may not use this file except in compliance with the License
|
||
|
* The Original Code is: vtiger CRM Open Source
|
||
|
* The Initial Developer of the Original Code is vtiger.
|
||
|
* Portions created by vtiger are Copyright (C) vtiger.
|
||
|
* All Rights Reserved.
|
||
|
******************************************************************************/
|
||
|
class VTTreeNode{
|
||
|
function __construct($arr){
|
||
|
$this->arr = $arr;
|
||
|
}
|
||
|
|
||
|
function getParams(){
|
||
|
$arr = $this->arr;
|
||
|
return array_slice($arr, 1, sizeof($arr)-1);
|
||
|
}
|
||
|
|
||
|
function getName(){
|
||
|
return $this->arr[0];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
class Symbol{
|
||
|
function __construct($value){
|
||
|
$this->value = $value;
|
||
|
}
|
||
|
|
||
|
function __toString(){
|
||
|
return "Symbol({$this->value})";
|
||
|
}
|
||
|
}
|
||
|
|
||
|
class VTParser{
|
||
|
function __construct($tokens){
|
||
|
$this->tokens = $tokens;
|
||
|
$this->tokenQueue = array();
|
||
|
}
|
||
|
|
||
|
function nextToken(){
|
||
|
if(sizeof($this->tokenQueue)==0){
|
||
|
return $this->tokens->nextToken();
|
||
|
}else{
|
||
|
return array_shift($this->tokenQueue);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function la($n = 1){
|
||
|
for($i=sizeof($this->tokenQueue); $i<$n; $i++){
|
||
|
$token = $this->tokens->nextToken();
|
||
|
$this->tokenQueue[] = $token;
|
||
|
}
|
||
|
return $this->tokenQueue[$n-1];
|
||
|
}
|
||
|
|
||
|
function consume($label, $value){
|
||
|
$token=$this->nextToken();
|
||
|
if($token->label!=$label || $token->value!=$value){
|
||
|
echo "Was expecting a $label of value $value got a {$token->label} of {$token->value} instead.";
|
||
|
throw new Exception("Was expecting a $label of value $value got a {$token->label} of {$token->value} instead.");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function consumeSymbol($sym){
|
||
|
$this->consume('SYMBOL', new Symbol($sym));
|
||
|
}
|
||
|
|
||
|
|
||
|
function check($token, $label, $value){
|
||
|
return $token->label == $label && $token->value==$value;
|
||
|
}
|
||
|
|
||
|
function checkSymbol($token, $sym){
|
||
|
return $this->check($token, 'SYMBOL', new Symbol($sym));
|
||
|
}
|
||
|
|
||
|
function atom(){
|
||
|
$token = $this->nextToken();
|
||
|
switch($token->label){
|
||
|
case "STRING":
|
||
|
return $token->value;
|
||
|
case "INTEGER":
|
||
|
return $token->value;
|
||
|
case "FLOAT":
|
||
|
return $token->value;
|
||
|
case "SYMBOL":
|
||
|
return $token->value;
|
||
|
case "OPEN_BRACKET":
|
||
|
$val = $this->expression();
|
||
|
$close = $this->nextToken();
|
||
|
if($close->label != 'CLOSE_BRACKET'){
|
||
|
throw new Exception("Was expecting a close bracket");
|
||
|
}
|
||
|
return $val;
|
||
|
default:
|
||
|
print_r($token);
|
||
|
throw new Exception();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function ifCondition(){
|
||
|
$this->consumeSymbol('if');
|
||
|
$cond = $this->expression();
|
||
|
$this->consumeSymbol('then');
|
||
|
$ifTrue = $this->expression();
|
||
|
$this->consumeSymbol('else');
|
||
|
if($this->checkSymbol($this->la(), 'if')){
|
||
|
$ifFalse = $this->ifCondition();
|
||
|
}else{
|
||
|
$ifFalse = $this->expression();
|
||
|
$this->consumeSymbol('end');
|
||
|
}
|
||
|
return new VTTreeNode(array(new Symbol('if'), $cond, $ifTrue, $ifFalse));
|
||
|
}
|
||
|
|
||
|
function expression(){
|
||
|
$la1 = $this->la(1);
|
||
|
$la2 = $this->la(2);
|
||
|
if($this->checkSymbol($la1, 'if')){
|
||
|
return $this->ifCondition();
|
||
|
}else if($la1->label=='SYMBOL' && $la2->label=='OPEN_BRACKET'){
|
||
|
$arr = array($this->nextToken()->value);
|
||
|
$this->nextToken();
|
||
|
if($this->la()->label != 'CLOSE_BRACKET'){
|
||
|
$arr[] = $this->expression();
|
||
|
$comma = $this->nextToken();
|
||
|
while($comma->label == 'COMMA'){
|
||
|
$arr[] = $this->expression();
|
||
|
$comma = $this->nextToken();
|
||
|
}
|
||
|
if($comma->label != 'CLOSE_BRACKET'){
|
||
|
throw new Exception("Was expecting a closing bracket");
|
||
|
}
|
||
|
}else{
|
||
|
$this->consume('CLOSE_BRACKET', new Symbol(')'));
|
||
|
}
|
||
|
return new VTTreeNode($arr);
|
||
|
}else{
|
||
|
return $this->binOp();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
var $precedence = array(
|
||
|
array('*', '/'),
|
||
|
array('+', '-'),
|
||
|
array('and', 'or'),
|
||
|
array('==', '>=', '<=', '>', '<')
|
||
|
);
|
||
|
|
||
|
function binOp(){
|
||
|
return $this->binOpPrec(sizeof($this->precedence)-1);
|
||
|
}
|
||
|
|
||
|
private function binOpPrec($prec){
|
||
|
if($prec>=0){
|
||
|
$lhs = $this->binOpPrec($prec-1);
|
||
|
$la = $this->la();
|
||
|
if($la->label == 'OPERATOR' && in_array($la->value->value, $this->precedence[$prec])){
|
||
|
$operator = $this->nextToken()->value;
|
||
|
$rhs = $this->binOpPrec($prec);
|
||
|
return new VTTreeNode(array($operator, $lhs, $rhs));
|
||
|
}else{
|
||
|
return $lhs;
|
||
|
}
|
||
|
}else{
|
||
|
return $this->unaryOp();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function unaryOp(){
|
||
|
$la = $this->la();
|
||
|
if($la->label=="OPERATOR" && in_array($la->value->value, array('+', '-'))){
|
||
|
$token = $this->nextToken();
|
||
|
$operator = $la->value;
|
||
|
$operand = $this->unaryOp();
|
||
|
return new VTTreeNode(array($operator, $operand));
|
||
|
}else{
|
||
|
return $this->atom();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
?>
|