4 ================================================================================
6 EvalMath - PHP Class to safely evaluate math expressions
7 Copyright (C) 2005 Miles Kaufmann <http://www.twmagic.com/>
9 ================================================================================
12 EvalMath - safely evaluate math expressions
16 include('evalmath.class.php');
19 $result = $m->evaluate('2+2');
20 // supports: order of operation; parentheses; negation; built-in functions
21 $result = $m->evaluate('-8(5/2)^2*(1-sqrt(4))-8');
22 // create your own variables
23 $m->evaluate('a = e^(ln(pi))');
25 $m->evaluate('f(x,y) = x^2 + y^2 - 2x*y + 1');
27 $result = $m->evaluate('3*f(42,a)');
31 Use the EvalMath class when you want to evaluate mathematical expressions
32 from untrusted sources. You can define your own variables and functions,
33 which are stored in the object. Try it, it's fun!
37 Evaluates the expression and returns the result. If an error occurs,
38 prints a warning and returns false. If $expr is a function assignment,
39 returns true on success.
42 A synonym for $m->evaluate().
45 Returns an associative array of all user-defined variables and values.
48 Returns an array of all user-defined functions.
52 Set to true to turn off warnings when evaluating expressions
55 If the last evaluation failed, contains a string describing the error.
56 (Useful when suppress_errors is on).
59 Copyright 2005, Miles Kaufmann.
62 Redistribution and use in source and binary forms, with or without
63 modification, are permitted provided that the following conditions are
66 1 Redistributions of source code must retain the above copyright
67 notice, this list of conditions and the following disclaimer.
68 2. Redistributions in binary form must reproduce the above copyright
69 notice, this list of conditions and the following disclaimer in the
70 documentation and/or other materials provided with the distribution.
71 3. The name of the author may not be used to endorse or promote
72 products derived from this software without specific prior written
75 THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
76 IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
77 WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
78 DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT,
79 INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
80 (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
81 SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
82 HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
83 STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
84 ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
85 POSSIBILITY OF SUCH DAMAGE.
90 * This class was heavily modified in order to get usefull spreadsheet emulation ;-)
97 var $suppress_errors = false;
98 var $last_error = null;
100 var $v = array(); // variables (and constants)
101 var $f = array(); // user-defined functions
102 var $vb = array(); // constants
103 var $fb = array( // built-in functions
104 'sin','sinh','arcsin','asin','arcsinh','asinh',
105 'cos','cosh','arccos','acos','arccosh','acosh',
106 'tan','tanh','arctan','atan','arctanh','atanh',
107 'sqrt','abs','ln','log','exp');
109 var $fc = array( // calc functions emulation
110 'average'=>array(-1), 'max'=>array(-1), 'min'=>array(-1),
111 'mod'=>array(2), 'pi'=>array(0), 'power'=>array(2),
112 'round'=>array(1, 2), 'sum'=>array(-1));
114 function EvalMath() {
118 return $this->evaluate($expr);
121 function evaluate($expr) {
122 $this->last_error
= null;
124 if (substr($expr, -1, 1) == ';') $expr = substr($expr, 0, strlen($expr)-1); // strip semicolons at the end
126 // is it a variable assignment?
127 if (preg_match('/^\s*([a-z][a-z0-9]*)\s*=\s*(.+)$/', $expr, $matches)) {
128 if (in_array($matches[1], $this->vb
)) { // make sure we're not assigning to a constant
129 return $this->trigger("cannot assign to constant '$matches[1]'");
131 if (($tmp = $this->pfx($this->nfx($matches[2]))) === false) return false; // get the result and make sure it's good
132 $this->v
[$matches[1]] = $tmp; // if so, stick it in the variable array
133 return $this->v
[$matches[1]]; // and return the resulting value
135 // is it a function assignment?
136 } elseif (preg_match('/^\s*([a-z][a-z0-9]*)\s*\(\s*([a-z][a-z0-9]*(?:\s*,\s*[a-z][a-z0-9]*)*)\s*\)\s*=\s*(.+)$/', $expr, $matches)) {
137 $fnn = $matches[1]; // get the function name
138 if (in_array($matches[1], $this->fb
)) { // make sure it isn't built in
139 return $this->trigger("cannot redefine built-in function '$matches[1]()'");
141 $args = explode(",", preg_replace("/\s+/", "", $matches[2])); // get the arguments
142 if (($stack = $this->nfx($matches[3])) === false) return false; // see if it can be converted to postfix
143 for ($i = 0; $i<count($stack); $i++
) { // freeze the state of the non-argument variables
145 if (preg_match('/^[a-z][a-z0-9]*$/', $token) and !in_array($token, $args)) {
146 if (array_key_exists($token, $this->v
)) {
147 $stack[$i] = $this->v
[$token];
149 return $this->trigger("undefined variable '$token' in function definition");
153 $this->f
[$fnn] = array('args'=>$args, 'func'=>$stack);
157 return $this->pfx($this->nfx($expr)); // straight up evaluation, woo
167 foreach ($this->f
as $fnn=>$dat)
168 $output[] = $fnn . '(' . implode(',', $dat['args']) . ')';
172 //===================== HERE BE INTERNAL METHODS ====================\\
174 // Convert infix to postfix notation
175 function nfx($expr) {
178 $stack = new EvalMathStack
;
179 $output = array(); // postfix form of expression, to be passed to pfx()
180 $expr = trim(strtolower($expr));
182 $ops = array('+', '-', '*', '/', '^', '_');
183 $ops_r = array('+'=>0,'-'=>0,'*'=>0,'/'=>0,'^'=>1); // right-associative operator?
184 $ops_p = array('+'=>0,'-'=>0,'*'=>1,'/'=>1,'_'=>1,'^'=>2); // operator precedence
186 $expecting_op = false; // we use this in syntax-checking the expression
187 // and determining when a - is a negation
189 if (preg_match("/[^\w\s+*^\/()\.,-]/", $expr, $matches)) { // make sure the characters are all good
190 return $this->trigger("illegal character '{$matches[0]}'");
193 while(1) { // 1 Infinite Loop ;)
194 $op = substr($expr, $index, 1); // get the first character at the current index
195 // find out if we're currently at the beginning of a number/variable/function/parenthesis/operand
196 $ex = preg_match('/^([a-z][a-z0-9]*\(?|\d+(?:\.\d*)?|\.\d+|\()/', substr($expr, $index), $match);
198 if ($op == '-' and !$expecting_op) { // is it a negation instead of a minus?
199 $stack->push('_'); // put a negation on the stack
201 } elseif ($op == '_') { // we have to explicitly deny this, because it's legal on the stack
202 return $this->trigger("illegal character '_'"); // but not in the input expression
204 } elseif ((in_array($op, $ops) or $ex) and $expecting_op) { // are we putting an operator on the stack?
205 if ($ex) { // are we expecting an operator but have a number/variable/function/opening parethesis?
206 return $this->trigger("expecting operand");
207 //$op = '*'; $index--; // it's an implicit multiplication
209 // heart of the algorithm:
210 while($stack->count
> 0 and ($o2 = $stack->last()) and in_array($o2, $ops) and ($ops_r[$op] ?
$ops_p[$op] < $ops_p[$o2] : $ops_p[$op] <= $ops_p[$o2])) {
211 $output[] = $stack->pop(); // pop stuff off the stack into the output
213 // many thanks: http://en.wikipedia.org/wiki/Reverse_Polish_notation#The_algorithm_in_detail
214 $stack->push($op); // finally put OUR operator onto the stack
216 $expecting_op = false;
218 } elseif ($op == ')' and $expecting_op) { // ready to close a parenthesis?
219 while (($o2 = $stack->pop()) != '(') { // pop off the stack back to the last (
220 if (is_null($o2)) return $this->trigger("unexpected ')'");
221 else $output[] = $o2;
223 if (preg_match("/^([a-z][a-z0-9]*)\($/", $stack->last(2), $matches)) { // did we just close a function?
224 $fnn = $matches[1]; // get the function name
225 $arg_count = $stack->pop(); // see how many arguments there were (cleverly stored on the stack, thank you)
227 $output[] = array('fn'=>$fn, 'fnn'=>$fnn, 'argcount'=>$arg_count); // send function to output
228 if (in_array($fnn, $this->fb
)) { // check the argument count
230 return $this->trigger("too many arguments ($arg_count given, 1 expected)");
231 } elseif (array_key_exists($fnn, $this->fc
)) {
232 $counts = $this->fc
[$fnn];
233 if (in_array(-1, $counts) and $arg_count > 0) {}
234 elseif (!in_array($arg_count, $counts))
235 return $this->trigger("wrong number of arguments ($arg_count given, " . implode('/',$this->fc
[$fnn]) . " expected)");
236 } elseif (array_key_exists($fnn, $this->f
)) {
237 if ($arg_count != count($this->f
[$fnn]['args']))
238 return $this->trigger("wrong number of arguments ($arg_count given, " . count($this->f
[$fnn]['args']) . " expected)");
239 } else { // did we somehow push a non-function on the stack? this should never happen
240 return $this->trigger("internal error");
245 } elseif ($op == ',' and $expecting_op) { // did we just finish a function argument?
246 while (($o2 = $stack->pop()) != '(') {
247 if (is_null($o2)) return $this->trigger("unexpected ','"); // oops, never had a (
248 else $output[] = $o2; // pop the argument expression stuff and push onto the output
250 // make sure there was a function
251 if (!preg_match("/^([a-z][a-z0-9]*)\($/", $stack->last(2), $matches))
252 return $this->trigger("unexpected ','");
253 $stack->push($stack->pop()+
1); // increment the argument count
254 $stack->push('('); // put the ( back on, we'll need to pop back to it again
256 $expecting_op = false;
258 } elseif ($op == '(' and !$expecting_op) {
259 $stack->push('('); // that was easy
263 } elseif ($ex and !$expecting_op) { // do we now have a function/variable/number?
264 $expecting_op = true;
266 if (preg_match("/^([a-z][a-z0-9]*)\($/", $val, $matches)) { // may be func, or variable w/ implicit multiplication against parentheses...
267 if (in_array($matches[1], $this->fb
) or array_key_exists($matches[1], $this->f
) or array_key_exists($matches[1], $this->fc
)) { // it's a func
271 $expecting_op = false;
272 } else { // it's a var w/ implicit multiplication
276 } else { // it's a plain old var or num
279 $index +
= strlen($val);
281 } elseif ($op == ')') {
282 //it could be only custom function with no params or general error
283 if ($stack->last() != '(' or $stack->last(2) != 1) return $this->trigger("unexpected ')'");
284 if (preg_match("/^([a-z][a-z0-9]*)\($/", $stack->last(3), $matches)) { // did we just close a function?
288 $fnn = $matches[1]; // get the function name
289 $counts = $this->fc
[$fnn];
290 if (!in_array(0, $counts))
291 return $this->trigger("wrong number of arguments ($arg_count given, " . implode('/',$this->fc
[$fnn]) . " expected)");
292 $output[] = array('fn'=>$fn, 'fnn'=>$fnn, 'argcount'=>0); // send function to output
295 return $this->trigger("unexpected ')'");
298 } elseif (in_array($op, $ops) and !$expecting_op) { // miscellaneous error checking
299 return $this->trigger("unexpected operator '$op'");
300 } else { // I don't even want to know what you did to get here
301 return $this->trigger("an unexpected error occured");
303 if ($index == strlen($expr)) {
304 if (in_array($op, $ops)) { // did we end with an operator? bad.
305 return $this->trigger("operator '$op' lacks operand");
310 while (substr($expr, $index, 1) == ' ') { // step the index past whitespace (pretty much turns whitespace
311 $index++
; // into implicit multiplication if no operator is there)
315 while (!is_null($op = $stack->pop())) { // pop everything off the stack and push onto output
316 if ($op == '(') return $this->trigger("expecting ')'"); // if there are (s on the stack, ()s were unbalanced
322 // evaluate postfix notation
323 function pfx($tokens, $vars = array()) {
325 if ($tokens == false) return false;
327 $stack = new EvalMathStack
;
329 foreach ($tokens as $token) { // nice and easy
331 // if the token is a function, pop arguments off the stack, hand them to the function, and push the result back on
332 if (is_array($token)) { // it's a function!
333 $fnn = $token['fnn'];
334 $count = $token['argcount'];
335 if (in_array($fnn, $this->fb
)) { // built-in function:
336 if (is_null($op1 = $stack->pop())) return $this->trigger("internal error");
337 $fnn = preg_replace("/^arc/", "a", $fnn); // for the 'arc' trig synonyms
338 if ($fnn == 'ln') $fnn = 'log';
339 eval('$stack->push(' . $fnn . '($op1));'); // perfectly safe eval()
340 } elseif (array_key_exists($fnn, $this->fc
)) { // calc emulation function
343 for ($i = $count-1; $i >= 0; $i--) {
344 if (is_null($args[] = $stack->pop())) return $this->trigger("internal error");
346 $res = call_user_func(array('EvalMathCalcEmul', $fnn), $args);
348 return $this->trigger("internal error");
351 } elseif (array_key_exists($fnn, $this->f
)) { // user function
354 for ($i = count($this->f
[$fnn]['args'])-1; $i >= 0; $i--) {
355 if (is_null($args[$this->f
[$fnn]['args'][$i]] = $stack->pop())) return $this->trigger("internal error");
357 $stack->push($this->pfx($this->f
[$fnn]['func'], $args)); // yay... recursion!!!!
359 // if the token is a binary operator, pop two values off the stack, do the operation, and push the result back on
360 } elseif (in_array($token, array('+', '-', '*', '/', '^'), true)) {
361 if (is_null($op2 = $stack->pop())) return $this->trigger("internal error");
362 if (is_null($op1 = $stack->pop())) return $this->trigger("internal error");
365 $stack->push($op1+
$op2); break;
367 $stack->push($op1-$op2); break;
369 $stack->push($op1*$op2); break;
371 if ($op2 == 0) return $this->trigger("division by zero");
372 $stack->push($op1/$op2); break;
374 $stack->push(pow($op1, $op2)); break;
376 // if the token is a unary operator, pop one value off the stack, do the operation, and push it back on
377 } elseif ($token == "_") {
378 $stack->push(-1*$stack->pop());
379 // if the token is a number or variable, push it on the stack
381 if (is_numeric($token)) {
382 $stack->push($token);
383 } elseif (array_key_exists($token, $this->v
)) {
384 $stack->push($this->v
[$token]);
385 } elseif (array_key_exists($token, $vars)) {
386 $stack->push($vars[$token]);
388 return $this->trigger("undefined variable '$token'");
392 // when we're out of tokens, the stack should have a single element, the final result
393 if ($stack->count
!= 1) return $this->trigger("internal error");
394 return $stack->pop();
397 // trigger an error, but nicely, if need be
398 function trigger($msg) {
399 $this->last_error
= $msg;
400 if (!$this->suppress_errors
) trigger_error($msg, E_USER_WARNING
);
406 class EvalMathStack
{
408 var $stack = array();
411 function push($val) {
412 $this->stack
[$this->count
] = $val;
417 if ($this->count
> 0) {
419 return $this->stack
[$this->count
];
424 function last($n=1) {
425 return $this->stack
[$this->count
-$n];
429 // spreadsheed functions emulation
430 // watch out for reversed args!!
431 class EvalMathCalcEmul
{
433 function average($args) {
434 return (EvalMathCalcEmul
::sum($args)/count($args));
437 function max($args) {
438 $res = array_pop($args);
439 foreach($args as $a) {
447 function min($args) {
448 $res = array_pop($args);
449 foreach($args as $a) {
457 function mod($args) {
458 return $args[1] %
$args[0];
465 function power($args) {
466 return $args[1]^
$args[0];
469 function round($args) {
470 if (count($args)==1) {
471 return round($args[0]);
473 return round($args[1], $args[0]);
477 function sum($args) {
479 foreach($args as $a) {