1
0

main.cc 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303
  1. //
  2. // main.cc
  3. //
  4. // Copyright (c) 2025 Yuji Hirose. All rights reserved.
  5. // MIT License
  6. //
  7. #include <atomic>
  8. #include <chrono>
  9. #include <ctime>
  10. #include <format>
  11. #include <iomanip>
  12. #include <iostream>
  13. #include <signal.h>
  14. #include <sstream>
  15. #include <httplib.h>
  16. using namespace httplib;
  17. const auto SERVER_NAME =
  18. std::format("cpp-httplib-server/{}", CPPHTTPLIB_VERSION);
  19. Server svr;
  20. void signal_handler(int signal) {
  21. if (signal == SIGINT || signal == SIGTERM) {
  22. std::cout << "\nReceived signal, shutting down gracefully...\n";
  23. svr.stop();
  24. }
  25. }
  26. std::string get_time_format() {
  27. auto now = std::chrono::system_clock::now();
  28. auto time_t = std::chrono::system_clock::to_time_t(now);
  29. std::stringstream ss;
  30. ss << std::put_time(std::localtime(&time_t), "%d/%b/%Y:%H:%M:%S %z");
  31. return ss.str();
  32. }
  33. std::string get_error_time_format() {
  34. auto now = std::chrono::system_clock::now();
  35. auto time_t = std::chrono::system_clock::to_time_t(now);
  36. std::stringstream ss;
  37. ss << std::put_time(std::localtime(&time_t), "%Y/%m/%d %H:%M:%S");
  38. return ss.str();
  39. }
  40. // NGINX Combined log format:
  41. // $remote_addr - $remote_user [$time_local] "$request" $status $body_bytes_sent
  42. // "$http_referer" "$http_user_agent"
  43. void nginx_access_logger(const Request &req, const Response &res) {
  44. std::string remote_user =
  45. "-"; // cpp-httplib doesn't have built-in auth user tracking
  46. auto time_local = get_time_format();
  47. auto request = std::format("{} {} {}", req.method, req.path, req.version);
  48. auto status = res.status;
  49. auto body_bytes_sent = res.body.size();
  50. auto http_referer = req.get_header_value("Referer");
  51. if (http_referer.empty()) http_referer = "-";
  52. auto http_user_agent = req.get_header_value("User-Agent");
  53. if (http_user_agent.empty()) http_user_agent = "-";
  54. std::cout << std::format("{} - {} [{}] \"{}\" {} {} \"{}\" \"{}\"",
  55. req.remote_addr, remote_user, time_local, request,
  56. status, body_bytes_sent, http_referer,
  57. http_user_agent)
  58. << std::endl;
  59. }
  60. // NGINX Error log format:
  61. // YYYY/MM/DD HH:MM:SS [level] message, client: client_ip, request: "request",
  62. // host: "host"
  63. void nginx_error_logger(const Error &err, const Request *req) {
  64. auto time_local = get_error_time_format();
  65. std::string level = "error";
  66. if (req) {
  67. auto request =
  68. std::format("{} {} {}", req->method, req->path, req->version);
  69. auto host = req->get_header_value("Host");
  70. if (host.empty()) host = "-";
  71. std::cerr << std::format("{} [{}] {}, client: {}, request: "
  72. "\"{}\", host: \"{}\"",
  73. time_local, level, to_string(err),
  74. req->remote_addr, request, host)
  75. << std::endl;
  76. } else {
  77. // If no request context, just log the error
  78. std::cerr << std::format("{} [{}] {}", time_local, level, to_string(err))
  79. << std::endl;
  80. }
  81. }
  82. void print_usage(const char *program_name) {
  83. std::cout << "Usage: " << program_name << " [OPTIONS]" << std::endl;
  84. std::cout << std::endl;
  85. std::cout << "Options:" << std::endl;
  86. std::cout << " --host <hostname> Server hostname (default: localhost)"
  87. << std::endl;
  88. std::cout << " --port <port> Server port (default: 8080)"
  89. << std::endl;
  90. std::cout << " --mount <mount:path> Mount point and document root"
  91. << std::endl;
  92. std::cout << " Format: mount_point:document_root"
  93. << std::endl;
  94. std::cout << " (default: /:./html)" << std::endl;
  95. std::cout << " --trusted-proxy <ip> Add trusted proxy IP address"
  96. << std::endl;
  97. std::cout << " (can be specified multiple times)"
  98. << std::endl;
  99. std::cout << " --version Show version information"
  100. << std::endl;
  101. std::cout << " --help Show this help message" << std::endl;
  102. std::cout << std::endl;
  103. std::cout << "Examples:" << std::endl;
  104. std::cout << " " << program_name
  105. << " --host localhost --port 8080 --mount /:./html" << std::endl;
  106. std::cout << " " << program_name
  107. << " --host 0.0.0.0 --port 3000 --mount /api:./api" << std::endl;
  108. std::cout << " " << program_name
  109. << " --trusted-proxy 192.168.1.100 --trusted-proxy 10.0.0.1"
  110. << std::endl;
  111. }
  112. struct ServerConfig {
  113. std::string hostname = "localhost";
  114. int port = 8080;
  115. std::string mount_point = "/";
  116. std::string document_root = "./html";
  117. std::vector<std::string> trusted_proxies;
  118. };
  119. enum class ParseResult { SUCCESS, HELP_REQUESTED, VERSION_REQUESTED, ERROR };
  120. ParseResult parse_command_line(int argc, char *argv[], ServerConfig &config) {
  121. for (int i = 1; i < argc; i++) {
  122. if (strcmp(argv[i], "--help") == 0 || strcmp(argv[i], "-h") == 0) {
  123. print_usage(argv[0]);
  124. return ParseResult::HELP_REQUESTED;
  125. } else if (strcmp(argv[i], "--host") == 0) {
  126. if (i + 1 >= argc) {
  127. std::cerr << "Error: --host requires a hostname argument" << std::endl;
  128. print_usage(argv[0]);
  129. return ParseResult::ERROR;
  130. }
  131. config.hostname = argv[++i];
  132. } else if (strcmp(argv[i], "--port") == 0) {
  133. if (i + 1 >= argc) {
  134. std::cerr << "Error: --port requires a port number argument"
  135. << std::endl;
  136. print_usage(argv[0]);
  137. return ParseResult::ERROR;
  138. }
  139. config.port = std::atoi(argv[++i]);
  140. if (config.port <= 0 || config.port > 65535) {
  141. std::cerr << "Error: Invalid port number. Must be between 1 and 65535"
  142. << std::endl;
  143. return ParseResult::ERROR;
  144. }
  145. } else if (strcmp(argv[i], "--mount") == 0) {
  146. if (i + 1 >= argc) {
  147. std::cerr
  148. << "Error: --mount requires mount_point:document_root argument"
  149. << std::endl;
  150. print_usage(argv[0]);
  151. return ParseResult::ERROR;
  152. }
  153. std::string mount_arg = argv[++i];
  154. auto colon_pos = mount_arg.find(':');
  155. if (colon_pos == std::string::npos) {
  156. std::cerr << "Error: --mount argument must be in format "
  157. "mount_point:document_root"
  158. << std::endl;
  159. print_usage(argv[0]);
  160. return ParseResult::ERROR;
  161. }
  162. config.mount_point = mount_arg.substr(0, colon_pos);
  163. config.document_root = mount_arg.substr(colon_pos + 1);
  164. if (config.mount_point.empty() || config.document_root.empty()) {
  165. std::cerr
  166. << "Error: Both mount_point and document_root must be non-empty"
  167. << std::endl;
  168. return ParseResult::ERROR;
  169. }
  170. } else if (strcmp(argv[i], "--version") == 0) {
  171. std::cout << CPPHTTPLIB_VERSION << std::endl;
  172. return ParseResult::VERSION_REQUESTED;
  173. } else if (strcmp(argv[i], "--trusted-proxy") == 0) {
  174. if (i + 1 >= argc) {
  175. std::cerr << "Error: --trusted-proxy requires an IP address argument"
  176. << std::endl;
  177. print_usage(argv[0]);
  178. return ParseResult::ERROR;
  179. }
  180. config.trusted_proxies.push_back(argv[++i]);
  181. } else {
  182. std::cerr << "Error: Unknown option '" << argv[i] << "'" << std::endl;
  183. print_usage(argv[0]);
  184. return ParseResult::ERROR;
  185. }
  186. }
  187. return ParseResult::SUCCESS;
  188. }
  189. bool setup_server(Server &svr, const ServerConfig &config) {
  190. svr.set_logger(nginx_access_logger);
  191. svr.set_error_logger(nginx_error_logger);
  192. // Set trusted proxies if specified
  193. if (!config.trusted_proxies.empty()) {
  194. svr.set_trusted_proxies(config.trusted_proxies);
  195. }
  196. auto ret = svr.set_mount_point(config.mount_point, config.document_root);
  197. if (!ret) {
  198. std::cerr
  199. << std::format(
  200. "Error: Cannot mount '{}' to '{}'. Directory may not exist.",
  201. config.mount_point, config.document_root)
  202. << std::endl;
  203. return false;
  204. }
  205. svr.set_file_extension_and_mimetype_mapping("html", "text/html");
  206. svr.set_file_extension_and_mimetype_mapping("htm", "text/html");
  207. svr.set_file_extension_and_mimetype_mapping("css", "text/css");
  208. svr.set_file_extension_and_mimetype_mapping("js", "text/javascript");
  209. svr.set_file_extension_and_mimetype_mapping("json", "application/json");
  210. svr.set_file_extension_and_mimetype_mapping("xml", "application/xml");
  211. svr.set_file_extension_and_mimetype_mapping("png", "image/png");
  212. svr.set_file_extension_and_mimetype_mapping("jpg", "image/jpeg");
  213. svr.set_file_extension_and_mimetype_mapping("jpeg", "image/jpeg");
  214. svr.set_file_extension_and_mimetype_mapping("gif", "image/gif");
  215. svr.set_file_extension_and_mimetype_mapping("svg", "image/svg+xml");
  216. svr.set_file_extension_and_mimetype_mapping("ico", "image/x-icon");
  217. svr.set_file_extension_and_mimetype_mapping("pdf", "application/pdf");
  218. svr.set_file_extension_and_mimetype_mapping("zip", "application/zip");
  219. svr.set_file_extension_and_mimetype_mapping("txt", "text/plain");
  220. svr.set_error_handler([](const Request & /*req*/, Response &res) {
  221. if (res.status == 404) {
  222. res.set_content(
  223. std::format(
  224. "<html><head><title>404 Not Found</title></head>"
  225. "<body><h1>404 Not Found</h1>"
  226. "<p>The requested resource was not found on this server.</p>"
  227. "<hr><p>{}</p></body></html>",
  228. SERVER_NAME),
  229. "text/html");
  230. }
  231. });
  232. svr.set_pre_routing_handler([](const Request & /*req*/, Response &res) {
  233. res.set_header("Server", SERVER_NAME);
  234. return Server::HandlerResponse::Unhandled;
  235. });
  236. signal(SIGINT, signal_handler);
  237. signal(SIGTERM, signal_handler);
  238. return true;
  239. }
  240. int main(int argc, char *argv[]) {
  241. ServerConfig config;
  242. auto result = parse_command_line(argc, argv, config);
  243. switch (result) {
  244. case ParseResult::HELP_REQUESTED:
  245. case ParseResult::VERSION_REQUESTED: return 0;
  246. case ParseResult::ERROR: return 1;
  247. case ParseResult::SUCCESS: break;
  248. }
  249. if (!setup_server(svr, config)) { return 1; }
  250. std::cout << "Serving HTTP on " << config.hostname << ":" << config.port
  251. << std::endl;
  252. std::cout << "Mount point: " << config.mount_point << " -> "
  253. << config.document_root << std::endl;
  254. if (!config.trusted_proxies.empty()) {
  255. std::cout << "Trusted proxies: ";
  256. for (size_t i = 0; i < config.trusted_proxies.size(); ++i) {
  257. if (i > 0) std::cout << ", ";
  258. std::cout << config.trusted_proxies[i];
  259. }
  260. std::cout << std::endl;
  261. }
  262. std::cout << "Press Ctrl+C to shutdown gracefully..." << std::endl;
  263. auto ret = svr.listen(config.hostname, config.port);
  264. std::cout << "Server has been shut down." << std::endl;
  265. return ret ? 0 : 1;
  266. }