ArgMap.h
1 /* Copyright (C) 2012-2020 IBM Corp.
2  * This program is Licensed under the Apache License, Version 2.0
3  * (the "License"); you may not use this file except in compliance
4  * with the License. You may obtain a copy of the License at
5  * http://www.apache.org/licenses/LICENSE-2.0
6  * Unless required by applicable law or agreed to in writing, software
7  * distributed under the License is distributed on an "AS IS" BASIS,
8  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
9  * See the License for the specific language governing permissions and
10  * limitations under the License. See accompanying LICENSE file.
11  */
12 
13 #ifndef HELIB_ARGMAP_H
14 #define HELIB_ARGMAP_H
15 
16 #include <iostream>
17 #include <iomanip>
18 #include <forward_list>
19 #include <initializer_list>
20 #include <set>
21 #include <unordered_map>
22 #include <algorithm>
23 #include <functional>
24 #include <string>
25 #include <sstream>
26 #include <fstream>
27 #include <cctype>
28 #include <memory>
29 #include <vector>
30 #include <tuple>
31 #include <type_traits>
32 #include <regex>
33 #include <helib/assertions.h>
34 
40 namespace helib {
41 
72 class ArgMap
73 {
74 private:
75  enum class ArgType
76  {
77  NAMED,
78  TOGGLE_TRUE,
79  TOGGLE_FALSE,
80  POSITIONAL,
81  DOTS
82  };
83 
84  // requires latching logic.
85  class PositionalArgsList
86  {
87  private:
88  std::vector<std::string> positional_args;
89  bool optional_flag = false;
90 
91  public:
92  void insert(std::string name, bool optional)
93  {
94  if (optional) {
95  this->optional_flag = true;
96  positional_args.push_back(name);
97  } else if (!optional && !optional_flag) {
98  positional_args.push_back(name);
99  } else {
100  throw LogicError("Attempting to have argument '" + name +
101  "' required after optional positional args given.");
102  }
103  }
104 
105  std::vector<std::string>::iterator begin()
106  {
107  return this->positional_args.begin();
108  }
109 
110  std::vector<std::string>::iterator end()
111  {
112  return this->positional_args.end();
113  }
114 
115  bool empty() { return this->positional_args.empty(); }
116  }; // end of PositionalArgsList
117 
118  struct ArgProcessor;
119 
120  template <typename C>
121  class ArgProcessorContainer;
122 
123  /* ArgProcessorValue: templated subclasses */
124  template <typename T>
125  class ArgProcessorValue;
126 
127 private:
128  ArgType arg_type = ArgType::NAMED;
129  char kv_separator = '=';
130  // Modes and other flags.
131  bool required_mode = false;
132  bool dots_enabled = false;
133  bool named_args_only = true;
134 
135  std::string progname;
136  std::string dots_name;
137 
138  // Track addresses to stop assigning same variable to more than one .arg(...)
139  std::set<void*> addresses_used;
140 
141  // Track what has been called previously whilst parsing.
142  std::set<std::string> previous_call_set;
143 
144  // Store the args.
145  std::unordered_map<std::string, std::shared_ptr<ArgProcessor>> map;
146 
147  // Docs held in vector until called by methods such as doc and usage
148  // Triple (arg name (+ value), docString, whether required)
149  std::vector<std::tuple<std::string, std::string, bool>> docVec;
150 
151  PositionalArgsList positional_args_list;
152 
153  std::unique_ptr<ArgProcessor> dots_ap;
154 
155  // Set for tracking required.
156  std::set<std::string> required_set;
157 
158  // Set for tracking optional.
159  std::set<std::string> optional_set;
160 
161  std::ostream* diagnostics_strm = nullptr;
162 
163  std::set<std::string> help_tokens = {"-h", "--help"};
164 
165  // Private for diagnostics
166  void printDiagnostics(const std::forward_list<std::string>& args) const;
167 
169 
175  void simpleParse(const std::forward_list<std::string>& args,
176  bool duplicates = true,
177  std::function<void(const std::string&)> stop = {});
178 
179 public:
180  enum class Separator
181  {
182  COLON,
183  EQUALS,
184  WHITESPACE
185  };
186 
198  template <typename T>
199  ArgMap& arg(const std::string& name, T& value);
200 
213  template <typename T>
214  ArgMap& arg(const std::string& name, T& value, const std::string& doc);
215 
229  template <typename T>
230  ArgMap& arg(const std::string& name,
231  T& value,
232  const std::string& doc,
233  const char* info);
234 
242  template <typename C>
243  ArgMap& dots(C& container, const char* name);
244 
253  ArgMap& parse(int argc, char** argv);
254 
263  ArgMap& parse(const std::string& filepath);
264 
270  ArgMap& optional();
271 
277  ArgMap& required();
278 
285  ArgMap& toggle(bool t = true);
286 
293  ArgMap& named();
294 
301  ArgMap& positional();
302 
308  ArgMap& helpArgs(const std::initializer_list<std::string> s);
309  ArgMap& helpArgs(const std::string s);
310 
317  ArgMap& diagnostics(std::ostream& ostrm = std::cout);
318 
326 
333  ArgMap& note(const std::string& s);
334 
340  void usage(const std::string& msg = "") const;
341 
347  std::string doc() const;
348 }; // End of class ArgMap
349 
350 // Three functions strip whitespaces before and after strings.
351 static inline void lstrip(std::string& s)
352 {
353  auto it = std::find_if(s.begin(), s.end(), [](unsigned char c) {
354  return !std::isspace(c);
355  });
356 
357  s.erase(s.begin(), it);
358 }
359 
360 static inline void rstrip(std::string& s)
361 {
362  auto it = std::find_if(s.rbegin(), s.rend(), [](unsigned char c) {
363  return !std::isspace(c);
364  });
365 
366  s.erase(it.base(), s.end());
367 }
368 
369 static inline void strip(std::string& s)
370 {
371  lstrip(s);
372  rstrip(s);
373 }
374 
375 // Correct the list from argv by splitting on the separator.
376 static void splitOnSeparator(std::forward_list<std::string>& args_lst, char sep)
377 {
378  if (sep == ' ')
379  return;
380 
381  for (auto it = args_lst.begin(); it != args_lst.end(); ++it) {
382  if (it->size() != 1) {
383  std::size_t pos = it->find(sep);
384  if (pos != std::string::npos) {
385  if (pos == 0) {
386  std::string sub = it->substr(1, std::string::npos);
387  *it = sep;
388  args_lst.insert_after(it, sub);
389  } else {
390  std::string sub = it->substr(pos);
391  *it = it->substr(0, pos);
392  args_lst.insert_after(it, sub);
393  }
394  }
395  }
396  }
397 }
398 
399 /* ArgProcessor: A virtual base class that acts as the interface to hold
400  * args in a map of different types.
401  */
403 {
404  virtual ~ArgProcessor() = default;
405  virtual ArgType getArgType() = 0;
406  virtual bool process(const std::string& s) = 0;
407 }; // end of ArgProcessor
408 
409 template <typename C>
410 class ArgMap::ArgProcessorContainer : public ArgProcessor
411 {
412 private:
413  C* container;
414  ArgType arg_type = ArgType::DOTS;
415 
416  using T = typename C::value_type;
417 
418  // For strings. Avoids a stream breaking on whitespace.
419  template <typename U = T,
420  typename S,
421  typename std::enable_if_t<std::is_same<U, S>::value, int> = 0>
422  bool do_process(const S& s)
423  {
424  container->push_back(s);
425  return true;
426  }
427 
428  // For non-string types, at the mercy of stringstream.
429  template <typename U = T,
430  typename S,
431  typename std::enable_if_t<!std::is_same<U, S>::value, int> = 0>
432  bool do_process(const S& s)
433  {
434  std::istringstream iss(s);
435  U tmp_value;
436  bool rt = (iss >> tmp_value);
437  container->push_back(tmp_value);
438  return rt;
439  }
440 
441 public:
442  ArgType getArgType() override { return arg_type; }
443 
444  bool process(const std::string& s) override { return this->do_process(s); }
445 
446  explicit ArgProcessorContainer(C* c) : container(c) {}
447 
448 }; // end of ArgProcessorContainer
449 
450 template <typename T>
451 class ArgMap::ArgProcessorValue : public ArgProcessor
452 {
453 private:
454  T* value;
455  ArgType arg_type;
456 
457  // For strings. Avoids a stream breaking on whitespace.
458  template <typename U = T,
459  typename S,
460  typename std::enable_if_t<std::is_same<U, S>::value, int> = 0>
461  bool do_process(const S& s)
462  {
463  *value = s;
464  return true;
465  }
466 
467  // For non-string types, at the mercy of stringstream.
468  template <typename U = T,
469  typename S,
470  typename std::enable_if_t<!std::is_same<U, S>::value, int> = 0>
471  bool do_process(const S& s)
472  {
473  std::istringstream iss(s);
474  return bool(iss >> *value);
475  }
476 
477 public:
478  ArgType getArgType() override { return arg_type; }
479 
480  bool process(const std::string& s) override { return this->do_process(s); }
481 
482  explicit ArgProcessorValue(T* v, ArgType at) : value(v), arg_type(at) {}
483 }; // end of ArgProcessorValue
484 
485 template <typename T>
486 ArgMap& ArgMap::arg(const std::string& name, T& value)
487 {
488  // trying to add empty or whitespace name?
489  assertTrue<LogicError>(
490  !name.empty() &&
491  std::none_of(name.begin(),
492  name.end(),
493  [](unsigned char c) { return std::isspace(c); }),
494  "Attempting to register an empty string or string with whitespace");
495 
496  // has this name already been added?
497  assertTrue<LogicError>(map.count(name) == 0,
498  "Key already in arg map (key: " + name + ")");
499 
500  // have we seen this addr before?
501  assertEq<LogicError>(addresses_used.count(&value),
502  0ul,
503  "Attempting to register variable twice");
504 
505  addresses_used.insert(&value);
506 
507  map[name] = std::make_shared<ArgProcessorValue<T>>(&value, this->arg_type);
508 
509  if (this->arg_type == ArgType::POSITIONAL) {
510  this->positional_args_list.insert(name, !this->required_mode);
511  }
512 
513  if (this->required_mode) {
514  // The user should make toggles correct in the code with optional
515  if (this->arg_type == ArgType::TOGGLE_TRUE ||
516  this->arg_type == ArgType::TOGGLE_FALSE)
517  throw LogicError("Toggle argument types cannot be required.");
518  this->required_set.insert(name);
519  } else {
520  // It is optional
521  this->optional_set.insert(name);
522  }
523 
524  return *this;
525 }
526 
527 template <typename T>
528 ArgMap& ArgMap::arg(const std::string& name, T& value, const std::string& doc)
529 {
530  arg(name, value);
531  std::stringstream ss;
532  ss << doc << " [ default=" << value << " ]";
533  docVec.push_back({name, ss.str(), required_mode});
534 
535  return *this;
536 }
537 
538 template <typename T>
539 ArgMap& ArgMap::arg(const std::string& name,
540  T& value,
541  const std::string& doc,
542  const char* info)
543 {
544  arg(name, value);
545 
546  docVec.push_back({name, doc, required_mode});
547  if (info != nullptr && info[0] != '\0')
548  std::get<1>(docVec.back()).append(" [ default=").append(info).append(" ]");
549 
550  return *this;
551 }
552 
553 template <typename C>
554 ArgMap& ArgMap::dots(C& container, const char* name)
555 {
556  if (this->dots_enabled)
557  throw LogicError(".dots() can only be called once.");
558 
559  this->dots_enabled = true;
560  this->dots_name = name;
561 
562  // Have it out of the map as it may be called many times and has no
563  // name/token.
564  this->dots_ap = std::make_unique<ArgProcessorContainer<C>>(&container);
565 
566  return *this;
567 }
568 
569 inline ArgMap& ArgMap::note(const std::string& s)
570 {
571  std::get<1>(docVec.back()).append("\t\t").append(s);
572  return *this;
573 }
574 
575 inline void ArgMap::usage(const std::string& msg) const
576 {
577  if (!msg.empty())
578  std::cerr << msg << '\n';
579 
580  decltype(docVec) docVecCopy = docVec;
581  std::stable_partition(
582  docVecCopy.begin(),
583  docVecCopy.end(),
584  [this](const auto& item) {
585  const auto& it = map.find(std::get<0>(item));
586  if (it == map.end())
587  throw LogicError("Not found in map '" + std::get<0>(item) + "'.");
588  else
589  return it->second->getArgType() != ArgType::POSITIONAL;
590  });
591 
592  std::cerr << "Usage: " << this->progname;
593  for (const auto& doc : docVecCopy) {
594  std::string name_ext(std::get<0>(doc));
595  auto it = map.find(name_ext);
596  if (it == map.end())
597  throw LogicError("Not found in map '" + name_ext + "'.");
598  if (it->second->getArgType() == ArgType::NAMED)
599  name_ext.append(1, this->kv_separator).append("<arg>");
600  if (std::get<2>(doc)) {
601  std::cerr << " " << name_ext;
602  } else {
603  std::cerr << " [" << name_ext << "]";
604  }
605  }
606 
607  if (this->dots_enabled)
608  std::cerr << " [" << this->dots_name << " ...]\n";
609  else
610  std::cerr << "\n";
611 
612  std::cerr << doc() << std::endl;
613 
614  exit(EXIT_FAILURE);
615 }
616 
617 inline ArgMap& ArgMap::helpArgs(const std::initializer_list<std::string> s)
618 {
619  this->help_tokens = s;
620  return *this;
621 }
622 
623 inline ArgMap& ArgMap::helpArgs(const std::string s)
624 {
625  this->help_tokens = {s};
626  return *this;
627 }
628 
630 {
631  switch (s) {
632  case Separator::EQUALS:
633  this->kv_separator = '=';
634  break;
635  case Separator::COLON:
636  this->kv_separator = ':';
637  break;
639  this->kv_separator = ' ';
640  break;
641  default:
642  // Use of class enums means it should never reach here.
643  throw LogicError("Unrecognised option for kv separator.");
644  }
645 
646  return *this;
647 }
648 
650 {
651  this->required_mode = false;
652  return *this;
653 }
654 
656 {
657  this->required_mode = true;
658  return *this;
659 }
660 
661 inline ArgMap& ArgMap::toggle(bool t)
662 {
663  this->named_args_only = false;
664  this->arg_type = t ? ArgType::TOGGLE_TRUE : ArgType::TOGGLE_FALSE;
665  return *this;
666 }
667 
669 {
670  this->arg_type = ArgType::NAMED;
671  return *this;
672 }
673 
675 {
676  this->named_args_only = false;
677  this->arg_type = ArgType::POSITIONAL;
678  return *this;
679 }
680 
681 inline ArgMap& ArgMap::diagnostics(std::ostream& ostrm)
682 {
683  this->diagnostics_strm = &ostrm;
684  return *this;
685 }
686 
687 inline void ArgMap::printDiagnostics(
688  const std::forward_list<std::string>& args) const
689 {
690  if (this->diagnostics_strm != nullptr) {
691  // argv as seen by ArgMap
692  *this->diagnostics_strm << "Args pre-parse:\n";
693  for (const auto& e : args) {
694  *this->diagnostics_strm << e << std::endl;
695  }
696  // required set
697  *this->diagnostics_strm << "Required args set:\n";
698  for (const auto& e : required_set) {
699  *this->diagnostics_strm << e << std::endl;
700  }
701  // optional set
702  *this->diagnostics_strm << "Optional args set:\n";
703  for (const auto& e : optional_set) {
704  *this->diagnostics_strm << e << std::endl;
705  }
706  }
707 }
708 
709 inline std::string ArgMap::doc() const
710 {
711  std::stringstream ss;
712  auto maxSzElem =
713  std::max_element(docVec.begin(),
714  docVec.end(),
715  [](const auto& x, const auto& y) {
716  return std::get<0>(x).size() < std::get<0>(y).size();
717  });
718 
719  for (const auto& p : docVec) {
720  ss << " " << std::left << std::setw(std::get<0>(*maxSzElem).length() + 1)
721  << std::get<0>(p) << std::setw(0) << std::get<1>(p) << '\n';
722  }
723 
724  return ss.str();
725 }
726 
727 inline void ArgMap::simpleParse(const std::forward_list<std::string>& args,
728  bool duplicates,
729  std::function<void(const std::string&)> stop)
730 {
731  if (stop == nullptr) {
732  stop = std::bind(&ArgMap::usage, this, std::placeholders::_1);
733  }
734 
735  auto pos_args_it = this->positional_args_list.begin();
736  for (auto it = args.begin(); it != args.end(); ++it) {
737 
738  const std::string token = *it;
739 
740  if (this->help_tokens.count(token))
741  stop("");
742 
743  // Check if not called before
744  if (!duplicates && this->previous_call_set.count(token))
745  stop("Attempting to set same variable '" + token + "' twice.");
746 
747  // Select ArgProcessor
748  auto map_it = this->map.find(token);
749  std::shared_ptr<ArgProcessor> ap =
750  (map_it == this->map.end()) ? nullptr : map_it->second;
751 
752  if (ap && ap->getArgType() != ArgType::POSITIONAL) {
753 
754  switch (ap->getArgType()) {
755  case ArgType::NAMED:
756  // Process value (parse and set)
757  if ((++it) == args.end())
758  stop("Dangling value for named argument '" + token + "'.");
759 
760  if (this->kv_separator == ' ') {
761  if (!ap->process(*it))
762  stop("Whitespace separator issue. Value:'" + *it + "'");
763  } else {
764  if ((++it) == args.end())
765  stop("Dangling value for named argument '" + token +
766  "' after separator.");
767  if (!ap->process(*it))
768  stop("Not a valid value '" + *it + "'.");
769  }
770  break;
771  case ArgType::TOGGLE_TRUE:
772  if (!ap->process("1"))
773  stop("");
774  break;
775  case ArgType::TOGGLE_FALSE:
776  if (!ap->process("0"))
777  stop("");
778  break;
779  default:
780  // Should never get here.
781  throw LogicError("Unrecognised ArgType.");
782  break;
783  }
784 
785  // Remove from required_set (if it is there)
786  this->required_set.erase(token);
787 
788  // Previously called.
789  this->previous_call_set.insert(token);
790  } else if (pos_args_it != this->positional_args_list.end()) {
791  // POSITIONAL args are treated differently as it is technically
792  // never a recognised token.
793  std::shared_ptr<ArgProcessor> pos_ap = map.find(*pos_args_it)->second;
794  if (!pos_ap->process(*it))
795  throw LogicError("Positional name does not match a ArgMap name.");
796  // Remove from required_set (if it is there)
797  this->required_set.erase(*pos_args_it);
798  ++pos_args_it;
799  } else if (this->dots_enabled) {
800  this->dots_ap->process(token);
801  } else {
802  std::string msg = "Unrecognised argument \'" + token + "\'";
803  if (!this->positional_args_list.empty())
804  msg += "\nThere could be too many positional arguments";
805  stop(msg);
806  }
807  }
808 }
809 
810 inline ArgMap& ArgMap::parse(int argc, char** argv)
811 {
812  this->progname = std::string(argv[0]);
813 
814  std::forward_list<std::string> args(argv + 1, argv + argc);
815 
816  splitOnSeparator(args, this->kv_separator);
817 
818  // Take any leading and trailing whitespace away.
819  std::for_each(args.begin(), args.end(), strip);
820 
821  printDiagnostics(args);
822 
823  simpleParse(args);
824 
825  // Have the required args been provided - if not exit
826  if (!this->required_set.empty()) {
827  std::ostringstream oss;
828  oss << "Required argument(s) not given:\n";
829  for (const auto& e : this->required_set)
830  oss << "\t" << e << '\n';
831  usage(oss.str()); // exits
832  }
833 
834  return *this;
835 }
836 
837 inline ArgMap& ArgMap::parse(const std::string& filepath)
838 {
839 
840  if (this->kv_separator == ' ') { // Not from files.
841  throw LogicError("Whitespace separator not possible from files.");
842  }
843 
844  if (!this->named_args_only) {
845  throw LogicError("Toggle and Positional arguments not possible from "
846  "files. Only named arguments.");
847  }
848 
849  std::ifstream file(filepath);
850  this->progname = filepath;
851 
852  if (!file.is_open()) {
853  throw RuntimeError("Could not open file '" + filepath + "'.");
854  }
855 
856  std::forward_list<std::string> args;
857  auto it = args.before_begin();
858  std::regex re_comment_lines(R"((^\s*\#)|(^\s+$))");
859  std::string line;
860  while (getline(file, line)) {
861  if (line.empty() || std::regex_search(line, re_comment_lines)) {
862  continue; // ignore comment lines and empties.
863  }
864  it = args.insert_after(it, line);
865  }
866 
867  splitOnSeparator(args, this->kv_separator);
868 
869  // Take any leading and trailing whitespace away.
870  std::for_each(args.begin(), args.end(), strip);
871 
872  printDiagnostics(args);
873 
874  simpleParse(args, false, [&filepath](const std::string& msg) {
875  throw RuntimeError("Could not parse params file: '" + filepath + "'. " +
876  msg);
877  });
878 
879  // Have the required args been provided - if not throw
880  if (!this->required_set.empty()) {
881  std::ostringstream oss;
882  oss << "Required argument(s) not given:\n";
883  for (const auto& e : this->required_set)
884  oss << "\t" << e << '\n';
885  throw RuntimeError(oss.str());
886  }
887 
888  return *this;
889 }
890 
891 } // namespace helib
892 
893 #endif // ifndef HELIB_ARGMAP_H
ArgMap & note(const std::string &s)
Adds a note to usage Adds a note to the arg usage description.
Definition: ArgMap.h:569
virtual bool process(const std::string &s)=0
Inherits from Exception and std::runtime_error.
Definition: exceptions.h:105
bool empty(const IndexSet &s)
Definition: IndexSet.h:182
void usage(const std::string &msg="") const
Print usage and exit Prints the usage and exits the program.
Definition: ArgMap.h:575
ArgMap & required()
Swaps to required arg mode Swaps to required arg mode. Following arguments will be considered require...
Definition: ArgMap.h:655
Inherits from Exception and std::logic_error.
Definition: exceptions.h:68
Definition: ArgMap.h:403
virtual ~ArgProcessor()=default
ArgMap & parse(int argc, char **argv)
Parse the argv array Parse the argv array If it fails or -h is an argument it prints the usage and ex...
Definition: ArgMap.h:810
void sub(const EncryptedArray &ea, PlaintextArray &pa, const PlaintextArray &other)
Definition: EncryptedArray.cpp:1033
ArgMap & separator(Separator s)
Sets the key-value separator Sets the named args key-value pair separator character.
Definition: ArgMap.h:629
args
Definition: gen-data.py:15
ArgMap & named()
Swaps to named arg type (default) Swaps to required arg mode. Following arguments will be considered ...
Definition: ArgMap.h:668
virtual ArgType getArgType()=0
Basic class for arg parsing. Example use:
Definition: ArgMap.h:73
ArgMap & diagnostics(std::ostream &ostrm=std::cout)
Turns on diagnostics printout when parsing Swaps to required arg mode. Following arguments will be co...
Definition: ArgMap.h:681
ArgMap & helpArgs(const std::initializer_list< std::string > s)
Provide custom help toggle args. (defaults are "-h", "--help") Overwrite default help toggle args to ...
Definition: ArgMap.h:617
ArgMap & toggle(bool t=true)
Swaps to toggle arg type Swaps to required arg mode. Following arguments will be considered of toggle...
Definition: ArgMap.h:661
Definition: apiAttributes.h:21
ArgMap & dots(C &container, const char *name)
Adds variable number of positional arg types after defined arg types are exhausted....
Definition: ArgMap.h:554
ArgMap & positional()
Swaps to positional arg type Swaps to required arg mode. Following arguments will be considered of po...
Definition: ArgMap.h:674
ArgMap & optional()
Swaps to optional arg mode (default) Swaps to optional arg mode. Following arguments will be consider...
Definition: ArgMap.h:649
ArgMap & arg(const std::string &name, T &value)
Add a new argument description Adds a new argument description with value of type T....
Definition: ArgMap.h:486
Separator
Definition: ArgMap.h:181
std::string doc() const
Return arg docs Returns the argument documentation as a string.
Definition: ArgMap.h:709
sep
Definition: gen-data.py:25