cli.h 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120
  1. //
  2. // cli.h
  3. // tax-manager
  4. //
  5. // Created by Sam Jaffe on 10/8/20.
  6. // Copyright © 2020 Sam Jaffe. All rights reserved.
  7. //
  8. #pragma once
  9. #include <functional>
  10. #include <iostream>
  11. #include <string>
  12. #include <tuple>
  13. #include <unordered_map>
  14. #include <vector>
  15. #include <string_utils/cast.h>
  16. #include <string_utils/tokenizer.h>
  17. #include "lambda_cast.h"
  18. #include "types.h"
  19. namespace cli {
  20. class cli {
  21. public:
  22. using args_t = std::vector<std::string>;
  23. using callback = std::function<void(args_t const &)>;
  24. private:
  25. string_utils::EscapedTokenizer tokenize_{" ", {'"', "\\\""}};
  26. std::istream *in_;
  27. std::string prompt_;
  28. std::unordered_map<std::string, size_t> arguments_;
  29. std::unordered_map<std::string, callback> callbacks_;
  30. mutable bool eof_;
  31. public:
  32. explicit cli(std::string const & prompt, std::istream &in = std::cin);
  33. explicit cli(std::istream &in = std::cin);
  34. template <typename... Args>
  35. cli & register_callback(std::string const & handle,
  36. std::function<void(Args...)> cb);
  37. template <typename F>
  38. cli & register_callback(std::string const & handle, F && cb) {
  39. return register_callback(handle, lambdas::FFL(cb));
  40. }
  41. template <typename T>
  42. cli & register_query(std::string const & handle, std::function<T()> cb);
  43. template <typename F>
  44. cli & register_query(std::string const & handle, F && cb) {
  45. return register_query(handle, lambdas::FFL(cb));
  46. }
  47. void run() const { in_ ? run_stream() : run_interactive(); }
  48. private:
  49. void run_interactive() const;
  50. void run_stream() const;
  51. void run(std::string &command) const;
  52. bool active() const;
  53. cli(cli const &) = delete;
  54. cli(cli &&) = delete;
  55. cli &operator=(cli const &) = delete;
  56. cli &operator=(cli &&) = delete;
  57. };
  58. template <typename... Args, size_t... Is>
  59. void cli_invoke(std::function<void(Args...)> cb, cli::args_t const & args,
  60. std::index_sequence<Is...>) {
  61. std::tuple<std::decay_t<Args>...> object;
  62. if (string_utils::cast(object, args)) {
  63. cb(std::move(std::get<Is>(object))...);
  64. }
  65. }
  66. template <typename... Args>
  67. cli & cli::register_callback(std::string const & handle,
  68. std::function<void(Args...)> cb) {
  69. constexpr size_t N = sizeof...(Args);
  70. arguments_.emplace(handle, N);
  71. callbacks_.emplace(handle, [=](args_t args) {
  72. if (N > args.size()) {
  73. std::cerr << "Missing Args in command '" << handle << "', " <<
  74. N << " required, but " << args.size() << " input\n";
  75. return;
  76. }
  77. args.resize(N);
  78. cli_invoke(cb, args, std::make_index_sequence<N>());
  79. });
  80. return *this;
  81. }
  82. template <typename T> void cli_print(T const & value) {
  83. if constexpr (detail::is_container_v<T> && !detail::is_string_v<T>) {
  84. using ::cli::cli_print;
  85. for (auto & elem : value) {
  86. cli_print(elem);
  87. }
  88. } else if constexpr (detail::is_map_value_type_v<T>) {
  89. std::cout << " " << value.first << ": " << value.second << "\n";
  90. } else {
  91. std::cout << " " << value << "\n";
  92. }
  93. }
  94. template <typename T>
  95. cli & cli::register_query(std::string const & handle, std::function<T()> cb) {
  96. arguments_.emplace(handle, 0);
  97. callbacks_.emplace(handle, [=](args_t const & args) {
  98. using ::cli::cli_print;
  99. cli_print(cb());
  100. });
  101. return *this;
  102. }
  103. }