ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
MathResultResolver.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
22 
33 
35 {
36  private \ilLanguage $lng;
37 
38  public function __construct(
39  private FieldSubstitution $substitution,
40  private Tokenizer $tokenizer,
41  ) {
42  }
43 
44  public function resolve(Token $token): Result
45  {
46  $calculated_token = $this->calculateFunctions($token);
47  $math_tokens = $this->tokenizer->tokenizeMath($calculated_token->getValue());
48 
49  $substituted = $this->substitution->substituteFieldValues($math_tokens);
50  [$result, $last_operator] = $this->parseMath($substituted);
51 
52  $from_function = $calculated_token->getFromFunction();
53 
54  if ($this->hasDateFieldsInMathTokens($math_tokens, $token)) {
55  return new DateResult(
56  (string) $result,
57  $from_function,
58  $last_operator
59  );
60  }
61 
62  return new IntegerResult(
63  (string) $result,
64  $from_function
65  );
66  }
67 
68  private function hasDateFieldsInMathTokens(array $math_tokens, Token $original_token): bool
69  {
70  foreach ($math_tokens as $math_token) {
71  if (str_starts_with($math_token->getValue(), Tokenizer::FIELD_OPENER)) {
72  $field = $this->substitution->getFieldFromPlaceholder($math_token->getValue());
73  if ($field->getDatatypeId() === \ilDclDatatype::INPUTFORMAT_DATETIME) {
74  return true;
75  }
76  }
77  }
78 
79  // fallback to original token
80  $tokens = $this->tokenizer->tokenize($original_token->getValue());
81  foreach ($tokens as $token) {
82  // find placeholders in token-values using regex, placeholders start with [[ and end with ]], there can be multiple
83  if (preg_match_all('/\[\[(.*?)\]\]/', $token->getValue(), $matches)) {
84  foreach ($matches[1] as $match) {
85  $field = $this->substitution->getFieldFromPlaceholder($match);
86  if ($field->getDatatypeId() === \ilDclDatatype::INPUTFORMAT_DATETIME) {
87  return true;
88  }
89  }
90  }
91  }
92 
93  return false;
94  }
95 
96  protected function parseMath(array $tokens): array
97  {
98  $operators = array_map(
99  static fn(Operators $operator): string => $operator->value,
101  );
102  $precedence = 0;
103  $stack = new Stack();
104  $precedences = new Stack();
105  $in_bracket = false;
106  foreach ($tokens as $token) {
107  if (is_string($token)) {
108  $token = new MathToken($token);
109  }
110 
111  // we use the tokens value
112  $token = $token->getValue() === '' ? '0' : $token->getValue();
113 
114  if (is_numeric($token) || $token === '(') {
115  $stack->push($token);
116  if ($token === '(') {
117  $in_bracket = true;
118  }
119  } elseif (in_array($token, $operators)) {
120  $last_operator = Operators::from($token);
121  $new_precedence = $last_operator->getPrecedence();
122  if ($new_precedence > $precedence || $in_bracket) {
123  // Precedence of operator is higher, push operator on stack
124  $stack->push($token);
125  $precedences->push($new_precedence);
126  $precedence = $new_precedence;
127  } else {
128  // Precedence is equal or lower, calculate result on stack
129  while ($new_precedence <= $precedence && $stack->count() > 1) {
130  $right = (float) $stack->pop();
131  $operator = $stack->pop();
132  $left = (float) $stack->pop();
133  $result = $this->calculate($operator, $left, $right);
134  $stack->push($result);
135  $precedence = $precedences->pop();
136  }
137  $stack->push($token);
138  $precedence = $new_precedence;
139  $precedences->push($new_precedence);
140  }
141  } elseif ($token === ')') {
142  // Need to calculate stack back to opening bracket
143  $_tokens = [];
144  $elem = $stack->pop();
145  while ($elem !== '(' && !$stack->isEmpty()) {
146  $_tokens[] = $elem;
147  $elem = $stack->pop();
148  }
149  // Get result within brackets recursive and push to stack
150  $stack->push($this->parseMath(array_reverse($_tokens)));
151  $in_bracket = false;
152  } else {
153  throw new \ilException("Unrecognized token '$token'");
154  }
155  }
156  // If one element is left on stack, we are done. Otherwise calculate
157  if ($stack->count() === 1) {
158  $result = $stack->pop();
159 
160  $value = (ctype_digit((string) $result)) ? $result : number_format($result, 2, '.', "'");
161  return [$value, $last_operator ?? null];
162  }
163 
164  while ($stack->count() >= 2) {
165  $right = $stack->pop();
166  $operator = $stack->pop();
167  $left = $stack->count() ? $stack->pop() : 0;
168  $stack->push($this->calculate($operator, $left, $right));
169  }
170  return [$stack->pop(), $last_operator];
171  }
172 
173  protected function calculateFunctions(Token $token): Token
174  {
175  $pattern = '#';
176  $functions = array_map(
177  static fn(Functions $function): string => $function->value,
179  );
180 
181  foreach ($functions as $function) {
182  $pattern .= "($function)\\(([^)]*)\\)|";
183  }
184  if (!preg_match_all(rtrim($pattern, '|') . '#', $token->getValue(), $result)) {
185  // No functions found inside token, just return token again
186  return $token;
187  }
188  $token_value = $token->getValue();
189  // Function found inside token, calculate!
190  foreach ($result[0] as $k => $to_replace) {
191  $function_args = $this->getFunctionArgs($k, $result);
192  $function = $function_args['function'];
193  $args = $this->substitution->substituteFieldValues($function_args['args']);
194  $token_value = str_replace($to_replace, (string) $this->calculateFunction($function, $args), $token_value);
195  }
196 
197  return new MathToken(
198  $token_value,
199  Functions::tryFrom($function ?? null)
200  );
201  }
202 
206  protected function getFunctionArgs(int $index, array $data): array
207  {
208  $return = [
209  'function' => '',
210  'args' => [],
211  ];
212  for ($i = 1; $i < count($data); $i++) {
213  $_data = $data[$i];
214  if ($_data[$index]) {
215  $function = $_data[$index];
216  $args = explode(';', $data[$i + 1][$index]);
217  $return['function'] = $function;
218  $return['args'] = $args;
219  break;
220  }
221  }
222 
223  return $return;
224  }
225 
226  protected function calculateFunction(string $function, array $args = [])
227  {
228  $args = array_map(function (Token $arg) {
229  return (float) $arg->getValue();
230  }, $args);
231 
232  switch ($function) {
233  case 'AVERAGE':
234  $count = count($args);
235  $array_sum = array_sum($args);
236 
237  return ($count > 0) ? $array_sum / $count : 0;
238  case 'SUM':
239  return array_sum($args);
240  case 'MIN':
241  return min($args);
242  case 'MAX':
243  return max($args);
244  default:
245  throw new ilException("Unrecognized function '$function'");
246  }
247  }
248 
256  protected function calculate(string $operator, $left, $right)
257  {
258  switch ($operator) {
259  case '+':
260  $result = $left + $right;
261  break;
262  case '-':
263  $result = $left - $right;
264  break;
265  case '*':
266  $result = $left * $right;
267  break;
268  case '/':
269  $result = ($right == 0) ? 0 : $left / $right;
270  break;
271  case '^':
272  $result = pow($left, $right);
273  break;
274  default:
275  throw new \ilException("Unrecognized operator '$operator'");
276  }
277 
278  return $result;
279  }
280 
281 }
Token
The string representation of these tokens must not occur in the names of metadata elements...
Definition: Token.php:27
__construct(private FieldSubstitution $substitution, private Tokenizer $tokenizer,)
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
$token
Definition: xapitoken.php:70
getFunctionArgs(int $index, array $data)
Helper method to return the function and its arguments from a preg_replace_all $result array...