/var/www/html/wp-content/plugins/woocommerce/includes/libraries/class-wc-eval-math.php


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
<?php

use Automattic\Jetpack\Constants;

if ( ! 
defined'ABSPATH' ) ) {
    exit;
}

if ( ! 
class_exists'WC_Eval_Math'false ) ) {
    
/**
     * Class WC_Eval_Math. Supports basic math only (removed eval function).
     *
     * Based on EvalMath by Miles Kaufman Copyright (C) 2005 Miles Kaufmann http://www.twmagic.com/.
     */
    
class WC_Eval_Math {

        
/**
         * Last error.
         *
         * @var string
         */
        
public static $last_error null;

        
/**
         * Variables (and constants).
         *
         * @var array
         */
        
public static $v = array( 'e' => 2.71'pi' => 3.14 );

        
/**
         * User-defined functions.
         *
         * @var array
         */
        
public static $f = array();

        
/**
         * Constants.
         *
         * @var array
         */
        
public static $vb = array( 'e''pi' );

        
/**
         * Built-in functions.
         *
         * @var array
         */
        
public static $fb = array();

        
/**
         * Evaluate maths string.
         *
         * @param string  $expr
         * @return mixed
         */
        
public static function evaluate$expr ) {
            
self::$last_error null;
            
$expr trim$expr );
            if ( 
substr$expr, -1) == ';' ) {
                
$expr substr$expr0strlen$expr ) -); // strip semicolons at the end
            
}
            
// ===============
            // is it a variable assignment?
            
if ( preg_match'/^\s*([a-z]\w*)\s*=\s*(.+)$/'$expr$matches ) ) {
                if ( 
in_array$matches[1], self::$vb ) ) { // make sure we're not assigning to a constant
                    
return self::trigger"cannot assign to constant '$matches[1]'" );
                }
                if ( ( 
$tmp self::pfxself::nfx$matches[2] ) ) ) === false ) {
                    return 
false// get the result and make sure it's good
                
}
                
self::$v$matches[1] ] = $tmp// if so, stick it in the variable array
                
return self::$v$matches[1] ]; // and return the resulting value
                // ===============
                // is it a function assignment?
            
} elseif ( preg_match'/^\s*([a-z]\w*)\s*\(\s*([a-z]\w*(?:\s*,\s*[a-z]\w*)*)\s*\)\s*=\s*(.+)$/'$expr$matches ) ) {
                
$fnn $matches[1]; // get the function name
                
if ( in_array$matches[1], self::$fb ) ) { // make sure it isn't built in
                    
return self::trigger"cannot redefine built-in function '$matches[1]()'" );
                }
                
$args explode","preg_replace"/\s+/"""$matches[2] ) ); // get the arguments
                
if ( ( $stack self::nfx$matches[3] ) ) === false ) {
                    return 
false// see if it can be converted to postfix
                
}
                
$stack_size count$stack );
                for ( 
$i 0$i $stack_size$i++ ) { // freeze the state of the non-argument variables
                    
$token $stack$i ];
                    if ( 
preg_match'/^[a-z]\w*$/'$token ) and ! in_array$token$args ) ) {
                        if ( 
array_key_exists$tokenself::$v ) ) {
                            
$stack$i ] = self::$v$token ];
                        } else {
                            return 
self::trigger"undefined variable '$token' in function definition" );
                        }
                    }
                }
                
self::$f$fnn ] = array( 'args' => $args'func' => $stack );
                return 
true;
                
// ===============
            
} else {
                return 
self::pfxself::nfx$expr ) ); // straight up evaluation, woo
            
}
        }

        
/**
         * Convert infix to postfix notation.
         *
         * @param  string $expr
         *
         * @return array|string
         */
        
private static function nfx$expr ) {

            
$index 0;
            
$stack = new WC_Eval_Math_Stack;
            
$output = array(); // postfix form of expression, to be passed to pfx()
            
$expr trim$expr );

            
$ops   = array( '+''-''*''/''^''_' );
            
$ops_r = array( '+' => 0'-' => 0'*' => 0'/' => 0'^' => ); // right-associative operator?
            
$ops_p = array( '+' => 0'-' => 0'*' => 1'/' => 1'_' => 1'^' => ); // operator precedence

            
$expecting_op false// we use this in syntax-checking the expression
            // and determining when a - is a negation
            
if ( preg_match"/[^\w\s+*^\/()\.,-]/"$expr$matches ) ) { // make sure the characters are all good
                
return self::trigger"illegal character '{$matches[0]}'" );
            }

            while ( 
) { // 1 Infinite Loop ;)
                
$op substr$expr$index); // get the first character at the current index
                // find out if we're currently at the beginning of a number/variable/function/parenthesis/operand
                
$ex preg_match'/^([A-Za-z]\w*\(?|\d+(?:\.\d*)?|\.\d+|\()/'substr$expr$index ), $match );
                
// ===============
                
if ( '-' === $op and ! $expecting_op ) { // is it a negation instead of a minus?
                    
$stack->push'_' ); // put a negation on the stack
                    
$index++;
                } elseif ( 
'_' === $op ) { // we have to explicitly deny this, because it's legal on the stack
                    
return self::trigger"illegal character '_'" ); // but not in the input expression
                    // ===============
                
} elseif ( ( in_array$op$ops ) or $ex ) and $expecting_op ) { // are we putting an operator on the stack?
                    
if ( $ex ) { // are we expecting an operator but have a number/variable/function/opening parenthesis?
                        
$op '*';
                        
$index--; // it's an implicit multiplication
                    
}
                    
// heart of the algorithm:
                    
while ( $stack->count 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 ] ) ) {
                        
$output[] = $stack->pop(); // pop stuff off the stack into the output
                    
}
                    
// many thanks: https://en.wikipedia.org/wiki/Reverse_Polish_notation#The_algorithm_in_detail
                    
$stack->push$op ); // finally put OUR operator onto the stack
                    
$index++;
                    
$expecting_op false;
                    
// ===============
                
} elseif ( ')' === $op && $expecting_op ) { // ready to close a parenthesis?
                    
while ( ( $o2 $stack->pop() ) != '(' ) { // pop off the stack back to the last (
                        
if ( is_null$o2 ) ) {
                            return 
self::trigger"unexpected ')'" );
                        } else {
                            
$output[] = $o2;
                        }
                    }
                    if ( 
preg_match"/^([A-Za-z]\w*)\($/"$stack->last), $matches ) ) { // did we just close a function?
                        
$fnn $matches[1]; // get the function name
                        
$arg_count $stack->pop(); // see how many arguments there were (cleverly stored on the stack, thank you)
                        
$output[] = $stack->pop(); // pop the function and push onto the output
                        
if ( in_array$fnnself::$fb ) ) { // check the argument count
                            
if ( $arg_count ) {
                                return 
self::trigger"too many arguments ($arg_count given, 1 expected)" );
                            }
                        } elseif ( 
array_key_exists$fnnself::$f ) ) {
                            if ( 
countself::$f$fnn ]['args'] ) != $arg_count ) {
                                return 
self::trigger"wrong number of arguments ($arg_count given, " countself::$f$fnn ]['args'] ) . " expected)" );
                            }
                        } else { 
// did we somehow push a non-function on the stack? this should never happen
                            
return self::trigger"internal error" );
                        }
                    }
                    
$index++;
                    
// ===============
                
} elseif ( ',' === $op and $expecting_op ) { // did we just finish a function argument?
                    
while ( ( $o2 $stack->pop() ) != '(' ) {
                        if ( 
is_null$o2 ) ) {
                            return 
self::trigger"unexpected ','" ); // oops, never had a (
                        
} else {
                            
$output[] = $o2// pop the argument expression stuff and push onto the output
                        
}
                    }
                    
// make sure there was a function
                    
if ( ! preg_match"/^([A-Za-z]\w*)\($/"$stack->last), $matches ) ) {
                        return 
self::trigger"unexpected ','" );
                    }
                    
$stack->push$stack->pop() + ); // increment the argument count
                    
$stack->push'(' ); // put the ( back on, we'll need to pop back to it again
                    
$index++;
                    
$expecting_op false;
                    
// ===============
                
} elseif ( '(' === $op and ! $expecting_op ) {
                    
$stack->push'(' ); // that was easy
                    
$index++;
                    
// ===============
                
} elseif ( $ex and ! $expecting_op ) { // do we now have a function/variable/number?
                    
$expecting_op true;
                    
$val $match[1];
                    if ( 
preg_match"/^([A-Za-z]\w*)\($/"$val$matches ) ) { // may be func, or variable w/ implicit multiplication against parentheses...
                        
if ( in_array$matches[1], self::$fb ) or array_key_exists$matches[1], self::$f ) ) { // it's a func
                            
$stack->push$val );
                            
$stack->push);
                            
$stack->push'(' );
                            
$expecting_op false;
                        } else { 
// it's a var w/ implicit multiplication
                            
$val $matches[1];
                            
$output[] = $val;
                        }
                    } else { 
// it's a plain old var or num
                        
$output[] = $val;
                    }
                    
$index += strlen$val );
                    
// ===============
                
} elseif ( ')' === $op ) { // miscellaneous error checking
                    
return self::trigger"unexpected ')'" );
                } elseif ( 
in_array$op$ops ) and ! $expecting_op ) {
                    return 
self::trigger"unexpected operator '$op'" );
                } else { 
// I don't even want to know what you did to get here
                    
return self::trigger"an unexpected error occurred" );
                }
                if ( 
strlen$expr ) == $index ) {
                    if ( 
in_array$op$ops ) ) { // did we end with an operator? bad.
                        
return self::trigger"operator '$op' lacks operand" );
                    } else {
                        break;
                    }
                }
                while ( 
substr$expr$index) == ' ' ) { // step the index past whitespace (pretty much turns whitespace
                    
$index++;                             // into implicit multiplication if no operator is there)
                
}
            }
            while ( ! 
is_null$op $stack->pop() ) ) { // pop everything off the stack and push onto output
                
if ( '(' === $op ) {
                    return 
self::trigger"expecting ')'" ); // if there are (s on the stack, ()s were unbalanced
                
}
                
$output[] = $op;
            }
            return 
$output;
        }

        
/**
         * Evaluate postfix notation.
         *
         * @param  mixed $tokens
         * @param  array $vars
         *
         * @return mixed
         */
        
private static function pfx$tokens$vars = array() ) {
            if ( 
false == $tokens ) {
                return 
false;
            }
            
$stack = new WC_Eval_Math_Stack;

            foreach ( 
$tokens as $token ) { // nice and easy
                // if the token is a binary operator, pop two values off the stack, do the operation, and push the result back on
                
if ( in_array$token, array( '+''-''*''/''^' ) ) ) {
                    if ( 
is_null$op2 $stack->pop() ) ) {
                        return 
self::trigger"internal error" );
                    }
                    if ( 
is_null$op1 $stack->pop() ) ) {
                        return 
self::trigger"internal error" );
                    }
                    switch ( 
$token ) {
                        case 
'+':
                            
$stack->push$op1 $op2 );
                            break;
                        case 
'-':
                            
$stack->push$op1 $op2 );
                            break;
                        case 
'*':
                            
$stack->push$op1 $op2 );
                            break;
                        case 
'/':
                            if ( 
== $op2 ) {
                                return 
self::trigger'division by zero' );
                            }
                            
$stack->push$op1 $op2 );
                            break;
                        case 
'^':
                            
$stack->pushpow$op1$op2 ) );
                            break;
                    }
                    
// if the token is a unary operator, pop one value off the stack, do the operation, and push it back on
                
} elseif ( '_' === $token ) {
                    
$stack->push( -$stack->pop() );
                    
// if the token is a function, pop arguments off the stack, hand them to the function, and push the result back on
                
} elseif ( ! preg_match"/^([a-z]\w*)\($/"$token$matches ) ) {
                    if ( 
is_numeric$token ) ) {
                        
$stack->push$token );
                    } elseif ( 
array_key_exists$tokenself::$v ) ) {
                        
$stack->pushself::$v$token ] );
                    } elseif ( 
array_key_exists$token$vars ) ) {
                        
$stack->push$vars$token ] );
                    } else {
                        return 
self::trigger"undefined variable '$token'" );
                    }
                }
            }
            
// when we're out of tokens, the stack should have a single element, the final result
            
if ( != $stack->count ) {
                return 
self::trigger"internal error" );
            }
            return 
$stack->pop();
        }

        
/**
         * Trigger an error, but nicely, if need be.
         *
         * @param  string $msg
         *
         * @return bool
         */
        
private static function trigger$msg ) {
            
self::$last_error $msg;
            if ( ! 
Constants::is_true'DOING_AJAX' ) && Constants::is_true'WP_DEBUG' ) ) {
                echo 
"\nError found in:";
                
self::debugPrintCallingFunction();
                
trigger_error$msgE_USER_WARNING );
            }
            return 
false;
        }

        
/**
         * Prints the file name, function name, and
         * line number which called your function
         * (not this function, then one that  called
         * it to begin with)
         */
        
private static function debugPrintCallingFunction() {
            
$file 'n/a';
            
$func 'n/a';
            
$line 'n/a';
            
$debugTrace debug_backtrace();
            if ( isset( 
$debugTrace[1] ) ) {
                
$file $debugTrace[1]['file'] ? $debugTrace[1]['file'] : 'n/a';
                
$line $debugTrace[1]['line'] ? $debugTrace[1]['line'] : 'n/a';
            }
            if ( isset( 
$debugTrace[2] ) ) {
                
$func $debugTrace[2]['function'] ? $debugTrace[2]['function'] : 'n/a';
            }
            echo 
"\n$file$func$line\n";
        }
    }

    
/**
     * Class WC_Eval_Math_Stack.
     */
    
class WC_Eval_Math_Stack {

        
/**
         * Stack array.
         *
         * @var array
         */
        
public $stack = array();

        
/**
         * Stack counter.
         *
         * @var integer
         */
        
public $count 0;

        
/**
         * Push value into stack.
         *
         * @param  mixed $val
         */
        
public function push$val ) {
            
$this->stack$this->count ] = $val;
            
$this->count++;
        }

        
/**
         * Pop value from stack.
         *
         * @return mixed
         */
        
public function pop() {
            if ( 
$this->count ) {
                
$this->count--;
                return 
$this->stack$this->count ];
            }
            return 
null;
        }

        
/**
         * Get last value from stack.
         *
         * @param  int $n
         *
         * @return mixed
         */
        
public function last$n=) {
            
$key $this->count $n;
            return 
array_key_exists$key$this->stack ) ? $this->stack$key ] : null;
        }
    }
}