The Battle for Wesnoth  1.19.10+dev
formula.cpp
Go to the documentation of this file.
1 /*
2  Copyright (C) 2003 - 2025
3  by David White <dave@whitevine.net>
4  Part of the Battle for Wesnoth Project https://www.wesnoth.org/
5 
6  This program is free software; you can redistribute it and/or modify
7  it under the terms of the GNU General Public License as published by
8  the Free Software Foundation; either version 2 of the License, or
9  (at your option) any later version.
10  This program is distributed in the hope that it will be useful,
11  but WITHOUT ANY WARRANTY.
12 
13  See the COPYING file for more details.
14 */
15 
16 #include "formula/formula.hpp"
17 
18 #include "formula/callable.hpp"
19 #include "formula/function.hpp"
20 #include "formula/string_utils.hpp"
21 #include "random.hpp"
23 #include "log.hpp"
24 
25 #include <cassert>
26 #include <set>
27 #include <sstream>
28 #include <utility>
29 
30 // This is here only for the below initialization code.
31 // If other logging is required in this file, it should use a different logdomain
32 // (probably "scripting/formula" or something)
33 static lg::log_domain log_engine("engine");
34 #define ERR_NG LOG_STREAM(err, log_engine)
35 
36 namespace utils {
37  namespace detail {
38  std::string evaluate_formula_impl(const std::string&);
39 
40  std::string evaluate_formula_impl(const std::string& formula) {
41  try {
42  const wfl::formula form(formula);
43  return form.evaluate().string_cast();
44  } catch(const wfl::formula_error& e) {
45  ERR_NG << "Formula in WML string cannot be evaluated due to "
46  << e.type << "\n\t--> \"";
47  return "";
48  }
49  }
50 
51  struct formula_initer {
54  }
55  } init;
56  }
57 }
58 
59 namespace wfl
60 {
61 using expr_table = std::map<std::string, expression_ptr>;
62 using expr_table_evaluated = std::map<std::string, variant>;
63 using expr_table_ptr = std::shared_ptr<expr_table>;
64 
65 // Function used when creating error reports.
66 // Parses all tokens passed to parse_expression, thus there are no EOL or whitespaces
67 static std::string tokens_to_string(const tk::token* i1, const tk::token* i2)
68 {
69  std::ostringstream expr;
70  while(i1 != i2) {
71  expr << std::string(i1->begin, i1->end) << " ";
72  ++i1;
73  }
74 
75  return expr.str();
76 }
77 
79 {
80 public:
82 
83  std::string str() const
84  {
85  return "";
86  }
87 
88 private:
89  variant execute(const formula_callable& /*variables*/, formula_debugger* /*fdb*/) const
90  {
91  return variant();
92  }
93 };
94 
95 // Implemented further down
96 expression_ptr parse_expression(const tk::token* i1, const tk::token* i2, function_symbol_table* symbols);
97 
98 
99 const char* const formula::id_chars = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ_";
100 
101 formula::formula(const std::string& text, function_symbol_table* symbols, bool manage_symbols)
102  : expr_()
103  , str_(text)
104  , managed_symbols_(symbols ? (manage_symbols ? symbols : nullptr) : new function_symbol_table)
105  , symbols_(symbols ? symbols : managed_symbols_.get())
106 {
107  std::vector<tk::token> tokens;
108  std::string::const_iterator i1 = text.begin(), i2 = text.end();
109 
110  // Set true when 'fai' keyword is found
111  bool fai_keyword = false;
112 
113  // Set true when 'wfl' keyword is found
114  bool wfl_keyword = false;
115 
116  // Used to locally keep the track of which file we parse actually and in which line we are
117  std::vector<std::pair<std::string, int>> files;
118 
119  // Used as a source of strings - we point to these strings from tokens
120  std::set<std::string> filenames;
121 
122  files.emplace_back("formula", 1);
123  filenames.insert("formula");
124 
125  std::set<std::string>::iterator filenames_it = filenames.begin();
126 
127  while(i1 != i2) {
128  try {
129  tokens.push_back(tk::get_token(i1,i2));
130 
131  tk::token_type current_type = tokens.back().type;
132 
133  if(current_type == tk::token_type::whitespace) {
134  tokens.pop_back();
135  } else if(current_type == tk::token_type::comment) {
136  // Since we can have multiline comments, let's see how many EOL are within it
137  int counter = 0;
138 
139  std::string comment = std::string(tokens.back().begin, tokens.back().end);
140  for(const auto& str_it : comment) {
141  if(str_it == '\n') {
142  counter++;
143  }
144  }
145 
146  files.back().second += counter;
147  tokens.pop_back();
148  } else if(current_type == tk::token_type::eol) {
149  files.back().second++;
150  tokens.pop_back();
151  } else if((current_type == tk::token_type::keyword) && (std::string(tokens.back().begin, tokens.back().end) == "fai")) {
152  fai_keyword = true;
153  tokens.pop_back();
154  } else if((current_type == tk::token_type::keyword) && (std::string(tokens.back().begin, tokens.back().end) == "wfl")) {
155  wfl_keyword = true;
156  tokens.pop_back();
157  } else if((current_type == tk::token_type::keyword) && (std::string(tokens.back().begin, tokens.back().end) == "faiend")) {
158  if(files.size() > 1) {
159  files.pop_back();
160  filenames_it = filenames.find(files.back().first);
161 
162  tokens.pop_back();
163  } else {
164  throw formula_error("Unexpected 'faiend' found", "", "", 0);
165  }
166  } else if((current_type == tk::token_type::keyword) && (std::string(tokens.back().begin, tokens.back().end) == "wflend")) {
167  if(files.size() > 1) {
168  files.pop_back();
169  filenames_it = filenames.find(files.back().first);
170 
171  tokens.pop_back();
172  } else {
173  throw formula_error("Unexpected 'wflend' found", "", "", 0);
174  }
175  } else if(fai_keyword || wfl_keyword) {
176  if(current_type == tk::token_type::string_literal) {
177  std::string str = std::string(tokens.back().begin, tokens.back().end);
178  files.emplace_back(str , 1);
179 
180  auto [pos, success] = filenames.insert(str);
181 
182  if(success) {
183  filenames_it = pos;
184  } else {
185  if(fai_keyword) {
186  throw formula_error("Faifile already included", "fai" + str, "", 0);
187  } else {
188  throw formula_error("Wflfile already included", "wfl" + str, "", 0);
189  }
190  }
191 
192  tokens.pop_back();
193  fai_keyword = false;
194  wfl_keyword = false;
195  } else {
196  if(fai_keyword) {
197  throw formula_error("Expected string after the 'fai'", "fai", "", 0);
198  } else {
199  throw formula_error("Expected string after the 'wfl'", "wfl", "", 0);
200  }
201  }
202  } else {
203  // In every token not specified above, store line number and name of file it came from
204  tokens.back().filename = &(*filenames_it);
205  tokens.back().line_number = files.back().second;
206  }
207  } catch(const tk::token_error& e) {
208  // When we catch token error, we should write whole line in which error occurred,
209  // so we merge info from token and everything we had in the line so far
210  std::string str = "";
211  if(!tokens.empty()) {
212  tk::token* tok_it = &tokens[0] + tokens.size()-1;
213  while(( tok_it != &tokens[0] ) && (tok_it->line_number == tokens.back().line_number)) {
214  --tok_it;
215  }
216 
217  if(tok_it != &tokens[0] && tok_it != &tokens[0] + tokens.size() -1) {
218  ++tok_it;
219  }
220 
221  str = tokens_to_string(tok_it, &tokens[0] + tokens.size());
222  }
223 
224  throw formula_error(e.description_, str + e.formula_, *filenames_it, files.back().second);
225  }
226  }
227 
228  if(files.size() > 1) {
229  throw formula_error("Missing 'wflend', make sure each .wfl file ends with it", "", "", 0);
230  }
231 
232  if(!tokens.empty()) {
233  expr_ = parse_expression(&tokens[0], &tokens[0] + tokens.size(), symbols_);
234  } else {
235  expr_ = std::make_shared<null_expression>();
236  }
237 }
238 
240  : expr_()
241  , str_()
242  , managed_symbols_(symbols ? nullptr : new function_symbol_table)
243  , symbols_(symbols ? symbols : managed_symbols_.get())
244 {
245  if(i1 != i2) {
246  expr_ = parse_expression(i1, i2, symbols);
247  } else {
248  expr_ = std::make_shared<null_expression>();
249  }
250 }
251 
252 // Explicitly defaulted out-of-line destructor (see header comment)
253 formula::~formula() = default;
254 
256 {
257  if(str.empty()) {
258  return formula_ptr();
259  }
260 
261  return formula_ptr(new formula(str, symbols));
262 }
263 
265 {
266  try {
267  return expr_->evaluate(variables, fdb);
268  } catch(const type_error& e) {
269  PLAIN_LOG << "formula type error: " << e.message;
270  return variant();
271  }
272 }
273 
275 {
276  static map_formula_callable null_callable;
277  return execute(null_callable,fdb);
278 }
279 
280 
281 formula_error::formula_error(const std::string& type, const std::string& formula,
282  const std::string& file, int line)
283  : error()
284  , type(type)
285  , formula(formula)
286  , filename(file)
287  , line(line)
288 {
289  std::stringstream ss;
290  ss << "Formula error in " << filename << ":" << line
291  << "\nIn formula " << formula
292  << "\nError: " << type;
293  message = ss.str();
294 }
295 
296 
297 /**
298  * Classes that encapsulate and handle the various formula functions.
299  */
301 {
302 public:
304  : symbols_(symbols)
305  {}
306 
307  virtual std::string str() const
308  {
309  return "{function_list_expression()}";
310  }
311 
312 private:
313  variant execute(const formula_callable& /*variables*/, formula_debugger* /*fdb*/) const
314  {
315  std::vector<variant> res;
316  for(const std::string& fcn_name : symbols_->get_function_names()) {
317  res.emplace_back(fcn_name);
318  }
319 
320  return variant(res);
321  }
322 
324 };
325 
327 {
328 public:
329  explicit list_expression(const std::vector<expression_ptr>& items)
330  : items_(items)
331  {}
332 
333 private:
334  variant execute(const formula_callable& variables, formula_debugger*fdb) const
335  {
336  std::vector<variant> res;
337  res.reserve(items_.size());
338  for(const auto& i : items_) {
339  res.push_back(i->evaluate(variables, add_debug_info(fdb, 0, "[list element]")));
340  }
341 
342  return variant(res);
343  }
344 
345  std::vector<expression_ptr> items_;
346 
347  std::string str() const
348  {
349  std::stringstream s;
350  s << '[';
351  bool first_item = true;
352  for(expression_ptr a : items_) {
353  if(!first_item) {
354  s << ',';
355  } else {
356  first_item = false;
357  }
358  s << a->str();
359  }
360  s << ']';
361  return s.str();
362  }
363 };
364 
366 {
367 public:
368  explicit map_expression(const std::vector<expression_ptr>& items)
369  : items_(items)
370  {}
371 
372  virtual std::string str() const
373  {
374  std::stringstream s;
375  s << " [";
376  for(std::vector<expression_ptr>::const_iterator i = items_.begin(); (i != items_.end()) && (i + 1 != items_.end()) ; i += 2) {
377  if(i != items_.begin()) {
378  s << ", ";
379  }
380  s << (*i)->str();
381  s << " -> ";
382  s << (*(i+1))->str();
383  }
384  if(items_.empty()) {
385  s << "->";
386  }
387  s << " ]";
388  return s.str();
389  }
390 
391 private:
392  variant execute(const formula_callable& variables, formula_debugger*fdb) const
393  {
394  std::map<variant,variant> res;
395  for(std::vector<expression_ptr>::const_iterator i = items_.begin(); (i != items_.end()) && (i + 1 != items_.end()) ; i += 2) {
396  variant key = (*i)->evaluate(variables, add_debug_info(fdb, 0, "key ->"));
397  variant value = (*(i+1))->evaluate(variables, add_debug_info(fdb, 1, "-> value"));
398  res[key] = value;
399  }
400 
401  return variant(res);
402  }
403 
404  std::vector<expression_ptr> items_;
405 };
406 
408 {
409 public:
410  unary_operator_expression(const std::string& op, expression_ptr arg)
411  : op_(),op_str_(op)
412  , operand_(std::move(arg))
413  {
414  if(op == "not") {
415  op_ = NOT;
416  } else if(op == "-") {
417  op_ = SUB;
418  } else {
419  throw formula_error("Illegal unary operator: '" + op + "'" , "", "", 0);
420  }
421  }
422 
423  virtual std::string str() const
424  {
425  std::stringstream s;
426  s << op_str_ << '('<< operand_->str() << ')';
427  return s.str();
428  }
429 
430 private:
431  variant execute(const formula_callable& variables, formula_debugger*fdb) const
432  {
433  const variant res = operand_->evaluate(variables, add_debug_info(fdb, 0, op_str_ + " unary"));
434  switch(op_) {
435  case NOT:
436  return res.as_bool() ? variant(0) : variant(1);
437  case SUB:
438  default:
439  return -res;
440  }
441  }
442 
443  enum OP { NOT, SUB };
445  std::string op_str_;
447 };
448 
450 {
451 public:
452  explicit string_callable(const variant& string) : string_(string) {}
453 
455  {
456  add_input(inputs, "size");
457  add_input(inputs, "empty");
458  add_input(inputs, "char");
459  add_input(inputs, "word");
460  add_input(inputs, "item");
461  }
462 
463  variant get_value(const std::string& key) const
464  {
465  if(key == "size") {
466  return variant(string_.as_string().length());
467  } else if(key == "empty") {
468  return variant(string_.as_string().empty());
469  } else if(key == "char" || key == "chars") {
470  std::vector<variant> chars;
471  for(char c : string_.as_string()) {
472  chars.emplace_back(std::string(1, c));
473  }
474 
475  return variant(chars);
476  } else if(key == "word" || key == "words") {
477  std::vector<variant> words;
478  const std::string& str = string_.as_string();
479  std::size_t next_space = 0;
480  do {
481  std::size_t last_space = next_space;
482  next_space = str.find_first_of(" \t", next_space);
483  words.emplace_back(str.substr(last_space, next_space - last_space));
484  next_space = str.find_first_not_of(" \t", next_space);
485  } while(next_space != std::string::npos);
486 
487  return variant(words);
488  } else if(key == "item" || key == "items") {
489  std::vector<std::string> split = utils::parenthetical_split(string_.as_string(), ',');
490  std::vector<variant> items;
491  items.reserve(split.size());
492  for(const std::string& s : split) {
493  items.emplace_back(s);
494  }
495 
496  return variant(items);
497  }
498 
499  return variant();
500  }
501 
502 private:
504 };
505 
507 {
508 public:
509  explicit list_callable(const variant& list) : list_(list) {}
510 
512  {
517  }
518 
519  variant get_value(const std::string& key) const
520  {
521  if(key == "size") {
522  return variant(list_.num_elements());
523  } else if(key == "empty") {
524  return variant(list_.num_elements() == 0);
525  } else if(key == "first") {
526  if(list_.num_elements() > 0) {
527  return list_[0];
528  }
529 
530  return variant();
531  } else if(key == "last") {
532  if(list_.num_elements() > 0) {
533  return list_[list_.num_elements()-1];
534  }
535 
536  return variant();
537  }
538 
539  return variant();
540  }
541 
542 private:
544 };
545 
547 {
548 public:
549  explicit map_callable(const variant& map) : map_(map) {}
550 
552  {
555 
556  for(const auto& v : map_) {
557  // variant_iterator does not implement operator->,
558  // and to do so is notrivial since it returns temporaries for maps.
559  const variant& key_variant = v.get_member("key");
560  if(!key_variant.is_string()) {
561  continue;
562  }
563 
564  std::string key = key_variant.as_string();
565  bool valid = true;
566  for(char c : key) {
567  if(!isalpha(c) && c != '_') {
568  valid = false;
569  break;
570  }
571  }
572 
573  if(valid) {
574  add_input(inputs, key);
575  }
576  }
577  }
578 
579  variant get_value(const std::string& key) const
580  {
581  const variant key_variant(key);
582  if(map_.as_map().find(key_variant) != map_.as_map().end()) {
583  return map_[key_variant];
584  } else if(key == "size") {
585  return variant(map_.num_elements());
586  } else if(key == "empty") {
587  return variant(map_.num_elements() == 0);
588  }
589 
590  return variant();
591  }
592 
593 private:
595 };
596 
598 {
599 public:
600  dot_callable(const formula_callable &global, const formula_callable& local)
601  : global_(global), local_(local)
602  {}
603 
604 private:
606 
608  {
609  return local_.get_inputs(inputs);
610  }
611 
612  variant get_value(const std::string& key) const
613  {
614  variant v = local_.query_value(key);
615 
616  if( v == variant() )
617  return global_.query_value(key);
618  else
619  return v;
620  }
621 };
622 
624 {
625 public:
627  : left_(std::move(left)), right_(std::move(right))
628  {}
629 
630  std::string str() const
631  {
632  std::stringstream s;
633  s << left_->str() << "." << right_->str();
634  return s.str();
635  }
636 
637 private:
638  variant execute(const formula_callable& variables, formula_debugger*fdb) const
639  {
640  variant left = left_->evaluate(variables, add_debug_info(fdb,0,"left ."));
641  if(!left.is_callable()) {
642  if(left.is_list()) {
643  list_callable list_call(left);
644  dot_callable callable(variables, list_call);
645  return right_->evaluate(callable,fdb);
646  }
647 
648  if(left.is_map()) {
649  map_callable map_call(left);
650  dot_callable callable(variables, map_call);
651  return right_->evaluate(callable,fdb);
652  }
653 
654  if(left.is_string()) {
655  string_callable string_call(left);
656  dot_callable callable(variables, string_call);
657  return right_->evaluate(callable,fdb);
658  }
659 
660  return left;
661  }
662 
663  dot_callable callable(variables, *left.as_callable());
664  return right_->evaluate(callable, add_debug_info(fdb,1,". right"));
665  }
666 
668 };
669 
671 {
672 public:
674  : left_(std::move(left)), key_(std::move(key))
675  {}
676 
677  std::string str() const
678  {
679  std::stringstream s;
680  s << left_->str() << '[' << key_->str() << ']';
681  return s.str();
682  }
683 
684 private:
685  variant execute(const formula_callable& variables, formula_debugger*fdb) const
686  {
687  const variant left = left_->evaluate(variables, add_debug_info(fdb,0,"base[]"));
688  const variant key = key_->evaluate(variables, add_debug_info(fdb,1,"[index]"));
689  if(left.is_list() || left.is_map()) {
690  return left[key];
691  }
692 
693  return variant();
694  }
695 
697 };
698 
700 {
701 public:
702  operator_expression(const std::string& op, expression_ptr left, expression_ptr right)
703  : op_(OP(op[0])), op_str_(op), left_(std::move(left)), right_(std::move(right))
704  {
705  if(op == ">=") {
706  op_ = GTE;
707  } else if(op == "<=") {
708  op_ = LTE;
709  } else if(op == "!=") {
710  op_ = NEQ;
711  } else if(op == "and") {
712  op_ = AND;
713  } else if(op == "or") {
714  op_ = OR;
715  } else if(op == ".+") {
716  op_ = ADDL;
717  } else if(op == ".-") {
718  op_ = SUBL;
719  } else if(op == ".*") {
720  op_ = MULL;
721  } else if(op == "./") {
722  op_ = DIVL;
723  } else if(op == "..") {
724  op_ = OP_CAT;
725  } else if(op == "in") {
726  op_ = OP_IN;
727  }
728  }
729 
730  std::string str() const
731  {
732  std::stringstream s;
733  s << '(' << left_->str() << op_str_ << right_->str() << ')';
734  return s.str();
735  }
736 
737 private:
738  variant execute(const formula_callable& variables, formula_debugger*fdb) const
739  {
740  const variant left = left_->evaluate(variables, add_debug_info(fdb, 0, "left " + op_str_));
741  const variant right = right_->evaluate(variables, add_debug_info(fdb, 1, op_str_ + " right"));
742 
743  switch(op_) {
744  case AND:
745  return left.as_bool() == false ? left : right;
746  case OR:
747  return left.as_bool() ? left : right;
748  case ADD:
749  return left + right;
750  case SUB:
751  return left - right;
752  case MUL:
753  return left * right;
754  case DIV:
755  return left / right;
756  case POW:
757  return left ^ right;
758  case ADDL:
759  return left.list_elements_add(right);
760  case SUBL:
761  return left.list_elements_sub(right);
762  case MULL:
763  return left.list_elements_mul(right);
764  case DIVL:
765  return left.list_elements_div(right);
766  case OP_IN:
767  return variant(right.contains(left));
768  case OP_CAT:
769  return left.concatenate(right);
770  case EQ:
771  return left == right ? variant(1) : variant(0);
772  case NEQ:
773  return left != right ? variant(1) : variant(0);
774  case LTE:
775  return left <= right ? variant(1) : variant(0);
776  case GTE:
777  return left >= right ? variant(1) : variant(0);
778  case LT:
779  return left < right ? variant(1) : variant(0);
780  case GT:
781  return left > right ? variant(1) : variant(0);
782  case MOD:
783  return left % right;
784  case RAN:
785  return left.build_range(right);
786  case DICE:
787  return variant(dice_roll(left.as_int(), right.as_int()));
788  default:
789  PLAIN_LOG << "ERROR: Unimplemented operator!";
790  return variant();
791  }
792  }
793 
794  static int dice_roll(int num_rolls, int faces)
795  {
796  int res = 0;
797  while(faces > 0 && num_rolls-- > 0) {
798  res += randomness::generator->get_random_int(1, faces);
799  }
800 
801  return res;
802  }
803 
804  //In some cases a IN or CAT macros are defined.
805  enum OP { AND, OR, NEQ, LTE, GTE, OP_CAT, OP_IN, GT='>', LT='<', EQ='=', RAN='~',
806  ADD='+', SUB='-', MUL='*', DIV='/', ADDL, SUBL, MULL, DIVL, DICE='d', POW='^', MOD='%' };
807 
809  std::string op_str_;
811 };
812 
814 {
815 public:
817  : formula_callable(false)
818  , base_(base)
819  , table_(std::move(table))
820  , evaluated_table_()
821  , debugger_(fdb)
822  {
823  }
824 
825 private:
830 
832  {
833  for(expr_table::const_iterator i = table_->begin(); i != table_->end(); ++i) {
834  add_input(inputs, i->first);
835  }
836  }
837 
838  variant get_value(const std::string& key) const
839  {
840  expr_table::iterator i = table_->find(key);
841  if(i != table_->end()) {
842  expr_table_evaluated::const_iterator ev = evaluated_table_.find(key);
843  if(ev != evaluated_table_.end()) {
844  return ev->second;
845  }
846 
847  variant v = i->second->evaluate(base_, add_debug_info(debugger_, 0, "where[" + key + "]"));
848  evaluated_table_[key] = v;
849  return v;
850  }
851 
852  return base_.query_value(key);
853  }
854 };
855 
857 {
858 public:
860  : body_(std::move(body)), clauses_(std::move(clauses))
861  {}
862 
863  std::string str() const
864  {
865  std::stringstream s;
866  s << "{where:(";
867  s << body_->str();
868  for(const expr_table::value_type &a : *clauses_) {
869  s << ", [" << a.first << "] -> ["<< a.second->str()<<"]";
870  }
871  s << ")}";
872  return s.str();
873  }
874 
875 private:
878 
879  variant execute(const formula_callable& variables,formula_debugger*fdb) const
880  {
881  where_variables wrapped_variables(variables, clauses_, fdb);
882  return body_->evaluate(wrapped_variables, add_debug_info(fdb, 0, "... where"));
883  }
884 };
885 
886 
888 {
889 public:
890  explicit identifier_expression(const std::string& id) : id_(id) {}
891 
892  std::string str() const
893  {
894  return id_;
895  }
896 
897 private:
898  variant execute(const formula_callable& variables, formula_debugger* /*fdb*/) const
899  {
900  return variables.query_value(id_);
901  }
902 
903  std::string id_;
904 };
905 
907 {
908 public:
909  explicit integer_expression(int i) : i_(i) {}
910 
911  std::string str() const
912  {
913  std::stringstream s;
914  s << i_;
915  return s.str();
916  }
917 
918 private:
919  variant execute(const formula_callable& /*variables*/, formula_debugger* /*fdb*/) const
920  {
921  return variant(i_);
922  }
923 
924  int i_;
925 };
926 
928 {
929 public:
930  decimal_expression(int i, int f) : i_(i), f_(f) {}
931 
932  std::string str() const
933  {
934  std::stringstream s;
935  s << i_ << '.';
936  s.width(3);
937  s.fill('0');
938  s << f_;
939  return s.str();
940  }
941 
942 private:
943  variant execute(const formula_callable& /*variables*/, formula_debugger* /*fdb*/) const
944  {
945  return variant(i_ * 1000 + f_, variant::DECIMAL_VARIANT );
946  }
947 
948  int i_, f_;
949 };
950 
952 {
953 public:
954  explicit string_expression(std::string str)
955  : str_()
956  , subs_()
957  {
958  std::string::iterator i = str.begin();
959  while((i = std::find(i, str.end(), '[')) != str.end()) {
960  int bracket_depth = 0;
961  std::string::iterator j = i + 1;
962  while(j != str.end() && (bracket_depth > 0 || *j != ']')) {
963  if(*j == '[') {
964  bracket_depth++;
965  } else if(*j == ']' && bracket_depth > 0) {
966  bracket_depth--;
967  }
968  ++j;
969  }
970 
971  if(j == str.end()) {
972  break;
973  }
974 
975  const std::string formula_str(i+1, j);
976  const int pos = std::distance(str.begin(), i);
977  if(j - i == 2 && (i[1] == '(' || i[1] == '\'' || i[1] == ')')) {
978  // Bracket contained nothing but a quote or parenthesis.
979  // This means it was intended as a literal quote or square bracket.
980  i = str.erase(i);
981  if(*i == '(') {
982  *i = '[';
983  } else if(*i == ')') {
984  *i = ']';
985  }
986 
987  i = str.erase(i + 1);
988  continue;
989  } else {
990  i = str.erase(i, j+1);
991  }
992 
993  substitution sub;
994  sub.pos = pos;
995  try {
996  sub.calculation.reset(new formula(formula_str));
997  } catch(formula_error& e) {
998  e.filename += " - string substitution";
999  throw;
1000  }
1001 
1002  subs_.push_back(sub);
1003  }
1004 
1005  std::reverse(subs_.begin(), subs_.end());
1006 
1007  str_ = variant(str);
1008  }
1009 
1010  std::string str() const
1011  {
1012  std::string res = str_.as_string();
1013  int j = res.size() - 1;
1014 
1015  for(const auto& sub : subs_) {
1016  for(; j >= sub.pos && j >= 0; j--) {
1017  if(res[j] == '\'') {
1018  res.replace(j, 1, "[']");
1019  } else if(res[j] == '[') {
1020  res.replace(j, 1, "[(]");
1021  } else if(res[j] == ']') {
1022  res.replace(j, 1, "[)]");
1023  }
1024  }
1025 
1026  const std::string str = "[" + sub.calculation->str() + "]";
1027  res.insert(sub.pos, str);
1028  }
1029 
1030  for(; j >= 0; j--) {
1031  if(res[j] == '\'') {
1032  res.replace(j, 1, "[']");
1033  } else if(res[j] == '[') {
1034  res.replace(j, 1, "[(]");
1035  } else if(res[j] == ']') {
1036  res.replace(j, 1, "[)]");
1037  }
1038  }
1039 
1040  return "'" + res + "'";
1041  }
1042 
1043 private:
1044  variant execute(const formula_callable& variables, formula_debugger*fdb) const
1045  {
1046  if(subs_.empty()) {
1047  return str_;
1048  }
1049 
1050  std::string res = str_.as_string();
1051  for(std::size_t i = 0; i < subs_.size(); ++i) {
1052  const int j = subs_.size() - i - 1;
1053  const substitution& sub = subs_[i];
1054  add_debug_info(fdb, j, "[string subst]");
1055  const std::string str = sub.calculation->evaluate(variables,fdb).string_cast();
1056  res.insert(sub.pos, str);
1057  }
1058 
1059  return variant(res);
1060  }
1061 
1063  {
1065 
1066  int pos;
1068  };
1069 
1071  std::vector<substitution> subs_;
1072 };
1073 
1074 
1075 /**
1076  * Functions to handle the actual parsing of WFL.
1077  */
1078 static int operator_precedence(const tk::token& t)
1079 {
1080  static std::map<std::string,int> precedence_map;
1081  if(precedence_map.empty()) {
1082  int n = 0;
1083  precedence_map["not"] = ++n;
1084  precedence_map["where"] = ++n;
1085  precedence_map["or"] = ++n;
1086  precedence_map["and"] = ++n;
1087  precedence_map["="] = ++n;
1088  precedence_map["!="] = n;
1089  precedence_map["<"] = n;
1090  precedence_map[">"] = n;
1091  precedence_map["<="] = n;
1092  precedence_map[">="] = n;
1093  precedence_map["in"] = n;
1094  precedence_map["~"] = ++n;
1095  precedence_map["+"] = ++n;
1096  precedence_map["-"] = n;
1097  precedence_map[".."] = n;
1098  precedence_map["*"] = ++n;
1099  precedence_map["/"] = n;
1100  precedence_map["%"] = ++n;
1101  precedence_map["^"] = ++n;
1102  precedence_map["d"] = ++n;
1103  precedence_map["."] = ++n;
1104  }
1105 
1106  assert(precedence_map.count(std::string(t.begin, t.end)));
1107  return precedence_map[std::string(t.begin, t.end)];
1108 }
1109 
1110 static void parse_function_args(const tk::token* &i1, const tk::token* i2, std::vector<std::string>* res)
1111 {
1112  const tk::token* begin = i1, *end = i2; // These are used for error reporting
1113 
1114  if(i1->type == tk::token_type::lparens) {
1115  ++i1;
1116  } else {
1117  throw formula_error("Invalid function definition", tokens_to_string(begin,end - 1), *i1->filename, i1->line_number);
1118  }
1119 
1120  while((i1-> type != tk::token_type::rparens) && (i1 != i2)) {
1121  if(i1->type == tk::token_type::identifier) {
1122  if(std::string((i1+1)->begin, (i1+1)->end) == "*") {
1123  res->push_back(std::string(i1->begin, i1->end) + std::string("*"));
1124  ++i1;
1125  } else {
1126  res->push_back(std::string(i1->begin, i1->end));
1127  }
1128  } else if(i1->type == tk::token_type::comma) {
1129  //do nothing
1130  } else {
1131  throw formula_error("Invalid function definition", tokens_to_string(begin,end - 1), *i1->filename, i1->line_number);
1132  }
1133 
1134  ++i1;
1135  }
1136 
1137  if(i1->type != tk::token_type::rparens) {
1138  throw formula_error("Invalid function definition", tokens_to_string(begin,end - 1), *i1->filename, i1->line_number);
1139  }
1140 
1141  ++i1;
1142 }
1143 
1144 static void parse_args(const tk::token* i1, const tk::token* i2,
1145  std::vector<expression_ptr>* res, function_symbol_table* symbols)
1146 {
1147  int parens = 0;
1148  const tk::token* beg = i1;
1149  while(i1 != i2) {
1150  if(i1->type == tk::token_type::lparens || i1->type == tk::token_type::lsquare ) {
1151  ++parens;
1152  } else if(i1->type == tk::token_type::rparens || i1->type == tk::token_type::rsquare ) {
1153  --parens;
1154  } else if(i1->type == tk::token_type::comma && !parens) {
1155  res->push_back(parse_expression(beg, i1, symbols));
1156  beg = i1+1;
1157  }
1158 
1159  ++i1;
1160  }
1161 
1162  if(beg != i1) {
1163  res->push_back(parse_expression(beg, i1, symbols));
1164  }
1165 }
1166 
1167 static void parse_set_args(const tk::token* i1, const tk::token* i2,
1168  std::vector<expression_ptr>* res, function_symbol_table* symbols)
1169 {
1170  int parens = 0;
1171  bool check_pointer = false;
1172  const tk::token* beg = i1;
1173  const tk::token* begin = i1, *end = i2; // These are used for error reporting
1174  while(i1 != i2) {
1175  if(i1->type == tk::token_type::lparens || i1->type == tk::token_type::lsquare) {
1176  ++parens;
1177  } else if(i1->type == tk::token_type::rparens || i1->type == tk::token_type::rsquare) {
1178  --parens;
1179  } else if(i1->type == tk::token_type::pointer && !parens ) {
1180  if(!check_pointer) {
1181  check_pointer = true;
1182  res->push_back(parse_expression(beg, i1, symbols));
1183  beg = i1+1;
1184  } else {
1185  throw formula_error("Too many '->' operators found", tokens_to_string(begin,end - 1), *i1->filename, i1->line_number);
1186  }
1187  } else if(i1->type == tk::token_type::comma && !parens ) {
1188  if(check_pointer)
1189  check_pointer = false;
1190  else {
1191  throw formula_error("Expected comma, but '->' found", tokens_to_string(begin,end - 1), *i1->filename, i1->line_number);
1192  }
1193  res->push_back(parse_expression(beg, i1, symbols));
1194  beg = i1+1;
1195  }
1196 
1197  ++i1;
1198  }
1199 
1200  if(beg != i1) {
1201  res->push_back(parse_expression(beg, i1, symbols));
1202  }
1203 }
1204 
1205 static void parse_where_clauses(const tk::token* i1, const tk::token* i2, const expr_table_ptr& res, function_symbol_table* symbols)
1206 {
1207  int parens = 0;
1208  const tk::token* original_i1_cached = i1;
1209  const tk::token* beg = i1;
1210  const tk::token* begin = i1, *end = i2; // These are used for error reporting
1211  std::string var_name;
1212 
1213  while(i1 != i2) {
1214  if(i1->type == tk::token_type::lparens || i1->type == tk::token_type::lsquare) {
1215  ++parens;
1216  } else if(i1->type == tk::token_type::rparens || i1->type == tk::token_type::rsquare) {
1217  --parens;
1218  } else if(!parens) {
1219  if(i1->type == tk::token_type::comma) {
1220  if(var_name.empty()) {
1221  throw formula_error("There is 'where <expression>' but 'where name=<expression>' was needed",
1222  tokens_to_string(begin, end - 1), *i1->filename, i1->line_number);
1223  }
1224 
1225  (*res)[var_name] = parse_expression(beg, i1, symbols);
1226  beg = i1+1;
1227  var_name = "";
1228  } else if(i1->type == tk::token_type::operator_token) {
1229  std::string op_name(i1->begin, i1->end);
1230 
1231  if(op_name == "=") {
1232  if(beg->type != tk::token_type::identifier) {
1233  if(i1 == original_i1_cached) {
1234  throw formula_error("There is 'where <expression>' but 'where name=<expression>' was needed",
1235  tokens_to_string(begin, end - 1), *i1->filename, i1->line_number);
1236  } else {
1237  throw formula_error("There is 'where <expression>=<expression>' but 'where name=<expression>' was needed",
1238  tokens_to_string(begin, end - 1), *i1->filename, i1->line_number);
1239  }
1240  } else if(beg+1 != i1) {
1241  throw formula_error("There is 'where name <expression>=<expression>' but 'where name=<expression>' was needed",
1242  tokens_to_string(begin, end - 1), *i1->filename, i1->line_number);
1243  } else if(!var_name.empty()) {
1244  throw formula_error("There is 'where name=name=<expression>' but 'where name=<expression>' was needed",
1245  tokens_to_string(begin, end - 1), *i1->filename, i1->line_number);
1246  }
1247 
1248  var_name.insert(var_name.end(), beg->begin, beg->end);
1249  beg = i1+1;
1250  }
1251  }
1252  }
1253  ++i1;
1254  }
1255 
1256  if(beg != i1) {
1257  if(var_name.empty()) {
1258  throw formula_error("There is 'where <expression>' but 'where name=<expression>' was needed",
1259  tokens_to_string(begin, end - 1), *i1->filename, i1->line_number);
1260  }
1261 
1262  (*res)[var_name] = parse_expression(beg, i1, symbols);
1263  }
1264 }
1265 
1267 {
1268  if(i1 == i2) {
1269  throw formula_error("Empty expression", "", *i1->filename, i1->line_number);
1270  }
1271 
1272  std::unique_ptr<function_symbol_table> temp_functions;
1273  if(!symbols) {
1274  temp_functions.reset(new function_symbol_table(function_symbol_table::get_builtins()));
1275  symbols = temp_functions.get();
1276  }
1277 
1278  const tk::token* begin = i1, *end = i2; // These are used for error reporting
1279 
1280  if(i1->type == tk::token_type::keyword && (i1 + 1)->type == tk::token_type::identifier) {
1281  if(std::string(i1->begin, i1->end) == "def") {
1282  ++i1;
1283  const std::string formula_name = std::string(i1->begin, i1->end);
1284 
1285  std::vector<std::string> args;
1286  parse_function_args(++i1, i2, &args);
1287 
1288  const tk::token* beg = i1;
1289  while((i1 != i2) && (i1->type != tk::token_type::semicolon)) {
1290  ++i1;
1291  }
1292 
1293  const std::string precond = "";
1294  if(symbols == nullptr) {
1295  throw formula_error("Function symbol table required but not present", "",*i1->filename, i1->line_number);
1296  }
1297 
1298  symbols->add_function(formula_name,
1299  std::make_shared<user_formula_function>(
1300  formula_name, std::make_shared<const formula>(beg, i1, symbols),
1301  formula::create_optional_formula(precond, symbols), args
1302  )
1303  );
1304 
1305  if((i1 == i2) || (i1 == (i2-1))) {
1306  return std::make_shared<function_list_expression>(symbols);
1307  } else {
1308  return parse_expression((i1+1), i2, symbols);
1309  }
1310  }
1311  }
1312 
1313  int parens = 0;
1314  const tk::token* op = nullptr;
1315  bool operator_group = false;
1316 
1317  for(const tk::token* i = i1; i != i2; ++i) {
1318  if(i->type == tk::token_type::lparens || i->type == tk::token_type::lsquare) {
1319  ++parens;
1320  } else if(i->type == tk::token_type::rparens || i->type == tk::token_type::rsquare) {
1321  --parens;
1322  } else if(parens == 0 && i->type == tk::token_type::operator_token) {
1323  if((!operator_group ) && (op == nullptr || operator_precedence(*op) >= operator_precedence(*i))) {
1324  // Need special exception for exponentiation to be right-associative
1325  if(*i->begin != '^' || op == nullptr || *op->begin != '^') {
1326  op = i;
1327  }
1328  }
1329  operator_group = true;
1330  } else {
1331  operator_group = false;
1332  }
1333  }
1334 
1335  if(op == nullptr) {
1336  // There's a situation when i1+1 equals i2-1 (meanwhile iter length is 2, like `()`)
1337  // Resulting in empty expression.
1338  if(i1->type == tk::token_type::lparens && (i2 - 1)->type == tk::token_type::rparens) {
1339  if(i1 + 1 == i2 - 1) {
1340  throw formula_error("No expression between parentheses", "()", *i1->filename, i1->line_number);
1341  }
1342  return parse_expression(i1+1,i2-1,symbols);
1343  } else if((i2 - 1)->type == tk::token_type::rsquare) { // check if there is [ ] : either a list/map definition, or a operator
1344  // First, a special case for an empty map
1345  if(i2 - i1 == 3 && i1->type == tk::token_type::lsquare && (i1+1)->type == tk::token_type::pointer) {
1346  return std::make_shared<map_expression>(std::vector<expression_ptr>());
1347  }
1348 
1349  const tk::token* tok = i2-2;
1350  int square_parens = 0;
1351  bool is_map = false;
1352  while ((tok->type != tk::token_type::lsquare || square_parens) && tok != i1) {
1353  if(tok->type == tk::token_type::rsquare) {
1354  square_parens++;
1355  } else if(tok->type == tk::token_type::lsquare) {
1356  square_parens--;
1357  } else if((tok->type == tk::token_type::pointer) && !square_parens ) {
1358  is_map = true;
1359  }
1360  --tok;
1361  }
1362 
1363  if(tok->type == tk::token_type::lsquare) {
1364  if(tok == i1) {
1365  // Create a list or a map
1366  std::vector<expression_ptr> args;
1367 
1368  if( is_map ) {
1369  parse_set_args(i1+1, i2-1, &args, symbols);
1370  return std::make_shared<map_expression>(args);
1371  } else {
1372  parse_args(i1+1,i2-1,&args,symbols);
1373  return std::make_shared<list_expression>(args);
1374  }
1375  } else {
1376  // Execute operator [ ]
1377  try{
1378  return std::make_shared<square_bracket_expression>(
1379  parse_expression(i1, tok, symbols),
1380  parse_expression(tok + 1, i2 - 1, symbols)
1381  );
1382  } catch (const formula_error& e){
1383  throw formula_error( e.type, tokens_to_string(i1, i2-1), *i1->filename, i1->line_number );
1384  }
1385  }
1386  }
1387  } else if(i2 - i1 == 1) {
1388  if(i1->type == tk::token_type::keyword) {
1389  if(std::string(i1->begin, i1->end) == "functions") {
1390  return std::make_shared<function_list_expression>(symbols);
1391  }
1392  } else if(i1->type == tk::token_type::identifier) {
1393  return std::make_shared<identifier_expression>(std::string(i1->begin, i1->end));
1394  } else if(i1->type == tk::token_type::integer) {
1395  int n = std::stoi(std::string(i1->begin, i1->end));
1396  return std::make_shared<integer_expression>(n);
1397  } else if(i1->type == tk::token_type::decimal) {
1398  tk::iterator dot = i1->begin;
1399  while(*dot != '.') {
1400  ++dot;
1401  }
1402 
1403  int n = std::stoi(std::string(i1->begin,dot));
1404 
1405  tk::iterator literal_end = i1->end;
1406 
1407  if(literal_end - dot > 4) {
1408  literal_end = dot + 4;
1409  }
1410 
1411  ++dot;
1412 
1413  int f = 0;
1414 
1415  int multiplicator = 100;
1416  while(dot != literal_end) {
1417  f += (*dot - 48) * multiplicator;
1418  multiplicator /= 10;
1419  ++dot;
1420  }
1421 
1422  return std::make_shared<decimal_expression>(n, f);
1423  } else if(i1->type == tk::token_type::string_literal) {
1424  return std::make_shared<string_expression>(std::string(i1->begin + 1, i1->end - 1));
1425  }
1426  } else if(i1->type == tk::token_type::identifier &&
1427  (i1+1)->type == tk::token_type::lparens &&
1428  (i2-1)->type == tk::token_type::rparens)
1429  {
1430  const tk::token* function_call_begin = i1, *function_call_end = i2; // These are used for error reporting
1431  int nleft = 0, nright = 0;
1432  for(const tk::token* i = i1; i != i2; ++i) {
1433  if(i->type == tk::token_type::lparens) {
1434  ++nleft;
1435  } else if(i->type == tk::token_type::rparens) {
1436  ++nright;
1437  }
1438  }
1439 
1440  if(nleft == nright) {
1441  std::vector<expression_ptr> args;
1442  parse_args(i1+2,i2-1,&args,symbols);
1443  try{
1444  return symbols->create_function(std::string(i1->begin, i1->end),args);
1445  }
1446  catch(const formula_error& e) {
1447  throw formula_error(e.type, tokens_to_string(function_call_begin, function_call_end), *i1->filename, i1->line_number);
1448  }
1449  }
1450  }
1451 
1452  throw formula_error("Could not parse expression", tokens_to_string(i1, i2), *i1->filename, i1->line_number);
1453  }
1454 
1455  if(op + 1 == end) {
1456  throw formula_error("Expected another token", tokens_to_string(begin, end - 1), *op->filename, op->line_number);
1457  }
1458 
1459  if(op == i1) {
1460  try{
1461  return expression_ptr(
1462  new unary_operator_expression(std::string(op->begin, op->end), parse_expression(op + 1, i2 ,symbols)));
1463  } catch(const formula_error& e) {
1464  throw formula_error( e.type, tokens_to_string(begin,end - 1), *op->filename, op->line_number);
1465  }
1466  }
1467 
1468  const std::string op_name(op->begin,op->end);
1469 
1470  if(op_name == ".") {
1471  return expression_ptr(
1472  new dot_expression(
1473  parse_expression(i1, op, symbols),
1474  parse_expression(op + 1,i2, symbols)
1475  )
1476  );
1477  }
1478 
1479  if(op_name == "where") {
1480  expr_table_ptr table(new expr_table());
1481  parse_where_clauses(op+1, i2, table, symbols);
1482 
1483  return std::make_shared<where_expression>(parse_expression(i1, op, symbols), table);
1484  }
1485 
1486  return expression_ptr(
1487  new operator_expression(op_name,
1488  parse_expression(i1, op, symbols),
1489  parse_expression(op + 1, i2, symbols)
1490  )
1491  );
1492 }
1493 
1494 } // namespace wfl
double t
Definition: astarsearch.cpp:63
int get_random_int(int min, int max)
Definition: random.hpp:51
decimal_expression(int i, int f)
Definition: formula.cpp:930
std::string str() const
Definition: formula.cpp:932
variant execute(const formula_callable &, formula_debugger *) const
Definition: formula.cpp:943
void get_inputs(formula_input_vector &inputs) const
Definition: formula.cpp:607
variant get_value(const std::string &key) const
Definition: formula.cpp:612
const formula_callable & local_
Definition: formula.cpp:605
dot_callable(const formula_callable &global, const formula_callable &local)
Definition: formula.cpp:600
const formula_callable & global_
Definition: formula.cpp:605
expression_ptr left_
Definition: formula.cpp:667
variant execute(const formula_callable &variables, formula_debugger *fdb) const
Definition: formula.cpp:638
std::string str() const
Definition: formula.cpp:630
dot_expression(expression_ptr left, expression_ptr right)
Definition: formula.cpp:626
expression_ptr right_
Definition: formula.cpp:667
variant query_value(const std::string &key) const
Definition: callable.hpp:50
formula_input_vector inputs() const
Definition: callable.hpp:63
static void add_input(formula_input_vector &inputs, const std::string &key, formula_access access_type=formula_access::read_only)
Definition: callable.hpp:136
virtual void get_inputs(formula_input_vector &) const
Definition: callable.hpp:70
static variant evaluate(const const_formula_ptr &f, const formula_callable &variables, formula_debugger *fdb=nullptr, variant default_res=variant(0))
Definition: formula.hpp:48
formula(const std::string &str, function_symbol_table *symbols=nullptr, bool manage_symbols=false)
Definition: formula.cpp:101
expression_ptr expr_
Definition: formula.hpp:89
function_symbol_table * symbols_
Definition: formula.hpp:92
static formula_ptr create_optional_formula(const std::string &str, function_symbol_table *symbols=nullptr)
Definition: formula.cpp:255
const std::string & str() const
Definition: formula.hpp:81
variant execute(const formula_callable &variables, formula_debugger *fdb=nullptr) const
Definition: formula.cpp:264
static const char *const id_chars
Definition: formula.hpp:83
Classes that encapsulate and handle the various formula functions.
Definition: formula.cpp:301
function_list_expression(function_symbol_table *symbols)
Definition: formula.cpp:303
variant execute(const formula_callable &, formula_debugger *) const
Definition: formula.cpp:313
virtual std::string str() const
Definition: formula.cpp:307
function_symbol_table * symbols_
Definition: formula.cpp:323
static std::shared_ptr< function_symbol_table > get_builtins()
Definition: function.cpp:1581
expression_ptr create_function(const std::string &fn, const std::vector< expression_ptr > &args) const
Definition: function.cpp:1549
void add_function(const std::string &name, formula_function_ptr &&fcn)
Definition: function.cpp:1544
std::set< std::string > get_function_names() const
Definition: function.cpp:1567
identifier_expression(const std::string &id)
Definition: formula.cpp:890
std::string str() const
Definition: formula.cpp:892
variant execute(const formula_callable &variables, formula_debugger *) const
Definition: formula.cpp:898
variant execute(const formula_callable &, formula_debugger *) const
Definition: formula.cpp:919
std::string str() const
Definition: formula.cpp:911
void get_inputs(formula_input_vector &inputs) const
Definition: formula.cpp:511
list_callable(const variant &list)
Definition: formula.cpp:509
variant get_value(const std::string &key) const
Definition: formula.cpp:519
std::string str() const
Definition: formula.cpp:347
list_expression(const std::vector< expression_ptr > &items)
Definition: formula.cpp:329
variant execute(const formula_callable &variables, formula_debugger *fdb) const
Definition: formula.cpp:334
std::vector< expression_ptr > items_
Definition: formula.cpp:345
map_callable(const variant &map)
Definition: formula.cpp:549
variant get_value(const std::string &key) const
Definition: formula.cpp:579
void get_inputs(formula_input_vector &inputs) const
Definition: formula.cpp:551
std::vector< expression_ptr > items_
Definition: formula.cpp:404
map_expression(const std::vector< expression_ptr > &items)
Definition: formula.cpp:368
virtual std::string str() const
Definition: formula.cpp:372
variant execute(const formula_callable &variables, formula_debugger *fdb) const
Definition: formula.cpp:392
std::string str() const
Definition: formula.cpp:83
variant execute(const formula_callable &, formula_debugger *) const
Definition: formula.cpp:89
expression_ptr right_
Definition: formula.cpp:810
expression_ptr left_
Definition: formula.cpp:810
variant execute(const formula_callable &variables, formula_debugger *fdb) const
Definition: formula.cpp:738
std::string str() const
Definition: formula.cpp:730
static int dice_roll(int num_rolls, int faces)
Definition: formula.cpp:794
operator_expression(const std::string &op, expression_ptr left, expression_ptr right)
Definition: formula.cpp:702
square_bracket_expression(expression_ptr left, expression_ptr key)
Definition: formula.cpp:673
variant execute(const formula_callable &variables, formula_debugger *fdb) const
Definition: formula.cpp:685
std::string str() const
Definition: formula.cpp:677
string_callable(const variant &string)
Definition: formula.cpp:452
void get_inputs(formula_input_vector &inputs) const
Definition: formula.cpp:454
variant get_value(const std::string &key) const
Definition: formula.cpp:463
string_expression(std::string str)
Definition: formula.cpp:954
variant execute(const formula_callable &variables, formula_debugger *fdb) const
Definition: formula.cpp:1044
std::vector< substitution > subs_
Definition: formula.cpp:1071
std::string str() const
Definition: formula.cpp:1010
unary_operator_expression(const std::string &op, expression_ptr arg)
Definition: formula.cpp:410
virtual std::string str() const
Definition: formula.cpp:423
variant execute(const formula_callable &variables, formula_debugger *fdb) const
Definition: formula.cpp:431
int as_int() const
Definition: variant.cpp:291
bool is_map() const
Definition: variant.hpp:68
bool is_list() const
Definition: variant.hpp:66
variant concatenate(const variant &v) const
Definition: variant.cpp:560
const_formula_callable_ptr as_callable() const
Definition: variant.hpp:83
std::size_t num_elements() const
Definition: variant.cpp:267
@ DECIMAL_VARIANT
Definition: variant.hpp:31
variant get_member(const std::string &name) const
Definition: variant.cpp:276
variant list_elements_mul(const variant &v) const
Definition: variant.cpp:548
variant list_elements_add(const variant &v) const
Definition: variant.cpp:536
const std::string & as_string() const
Definition: variant.cpp:318
std::string string_cast() const
Definition: variant.cpp:638
variant list_elements_div(const variant &v) const
Definition: variant.cpp:554
bool as_bool() const
Returns a boolean state of the variant value.
Definition: variant.cpp:313
bool is_string() const
Definition: variant.hpp:67
variant build_range(const variant &v) const
Definition: variant.cpp:586
bool is_callable() const
Definition: variant.hpp:65
const std::map< variant, variant > & as_map() const
Definition: variant.cpp:330
variant list_elements_sub(const variant &v) const
Definition: variant.cpp:542
expression_ptr body_
Definition: formula.cpp:876
variant execute(const formula_callable &variables, formula_debugger *fdb) const
Definition: formula.cpp:879
std::string str() const
Definition: formula.cpp:863
expr_table_ptr clauses_
Definition: formula.cpp:877
where_expression(expression_ptr body, expr_table_ptr clauses)
Definition: formula.cpp:859
const formula_callable & base_
Definition: formula.cpp:826
variant get_value(const std::string &key) const
Definition: formula.cpp:838
where_variables(const formula_callable &base, expr_table_ptr table, formula_debugger *fdb)
Definition: formula.cpp:816
void get_inputs(formula_input_vector &inputs) const
Definition: formula.cpp:831
expr_table_ptr table_
Definition: formula.cpp:827
expr_table_evaluated evaluated_table_
Definition: formula.cpp:828
formula_debugger * debugger_
Definition: formula.cpp:829
static lg::log_domain log_engine("engine")
#define ERR_NG
Definition: formula.cpp:34
std::size_t i
Definition: function.cpp:1022
expression_ptr expr_
Definition: function.cpp:809
std::string id
Text to match against addon_info.tags()
Definition: manager.cpp:199
Standard logging facilities (interface).
#define PLAIN_LOG
Definition: log.hpp:296
CURSOR_TYPE get()
Definition: cursor.cpp:216
void line(int from_x, int from_y, int to_x, int to_y)
Draw a line.
Definition: draw.cpp:190
bool is_map(const std::string &filename)
Returns true if the file ends with the mapfile extension.
rng * generator
This generator is automatically synced during synced context.
Definition: random.cpp:60
boost::variant< constant, n_var, boost::recursive_wrapper< not_op >, boost::recursive_wrapper< ternary_op > > expr
struct utils::detail::formula_initer init
std::string(* evaluate_formula)(const std::string &formula)
std::string evaluate_formula_impl(const std::string &)
Definition: formula.cpp:40
constexpr auto reverse
Definition: ranges.hpp:40
int stoi(std::string_view str)
Same interface as std::stoi and meant as a drop in replacement, except:
Definition: charconv.hpp:154
std::vector< std::string > parenthetical_split(std::string_view val, const char separator, std::string_view left, std::string_view right, const int flags)
Splits a string based either on a separator, except then the text appears within specified parenthesi...
std::vector< std::string > split(const config_attribute_value &val)
auto * find(Container &container, const Value &value)
Convenience wrapper for using find on a container without needing to comare to end()
Definition: general.hpp:140
token get_token(iterator &i1, const iterator i2)
Definition: tokenizer.cpp:44
std::string::const_iterator iterator
Definition: tokenizer.hpp:25
Definition: contexts.hpp:43
formula_debugger * add_debug_info(formula_debugger *fdb, int arg_number, const std::string &f_name)
std::map< std::string, expression_ptr > expr_table
Definition: formula.cpp:61
static void parse_function_args(const tk::token *&i1, const tk::token *i2, std::vector< std::string > *res)
Definition: formula.cpp:1110
std::vector< formula_input > formula_input_vector
std::shared_ptr< const formula > const_formula_ptr
Definition: formula_fwd.hpp:24
std::shared_ptr< formula_expression > expression_ptr
Definition: formula.hpp:29
static int operator_precedence(const tk::token &t)
Functions to handle the actual parsing of WFL.
Definition: formula.cpp:1078
std::shared_ptr< expr_table > expr_table_ptr
Definition: formula.cpp:63
std::shared_ptr< formula > formula_ptr
Definition: formula_fwd.hpp:22
static void parse_set_args(const tk::token *i1, const tk::token *i2, std::vector< expression_ptr > *res, function_symbol_table *symbols)
Definition: formula.cpp:1167
std::map< std::string, variant > expr_table_evaluated
Definition: formula.cpp:62
expression_ptr parse_expression(const tk::token *i1, const tk::token *i2, function_symbol_table *symbols)
Definition: formula.cpp:1266
static std::string tokens_to_string(const tk::token *i1, const tk::token *i2)
Definition: formula.cpp:67
static void parse_where_clauses(const tk::token *i1, const tk::token *i2, const expr_table_ptr &res, function_symbol_table *symbols)
Definition: formula.cpp:1205
static void parse_args(const tk::token *i1, const tk::token *i2, std::vector< expression_ptr > *res, function_symbol_table *symbols)
Definition: formula.cpp:1144
std::string filename
Filename.
std::string message
Definition: exceptions.hpp:30
std::string type
Definition: formula.hpp:112
std::string filename
Definition: formula.hpp:114
const std::string * filename
Definition: tokenizer.hpp:69
mock_char c
static map_location::direction n
static map_location::direction s
#define e
#define f