HttpService.h 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245
  1. #ifndef HV_HTTP_SERVICE_H_
  2. #define HV_HTTP_SERVICE_H_
  3. #include <string>
  4. #include <map>
  5. #include <unordered_map>
  6. #include <list>
  7. #include <memory>
  8. #include <functional>
  9. #include "hexport.h"
  10. #include "HttpMessage.h"
  11. #include "HttpResponseWriter.h"
  12. #include "HttpContext.h"
  13. #define DEFAULT_BASE_URL "/api/v1"
  14. #define DEFAULT_DOCUMENT_ROOT "/var/www/html"
  15. #define DEFAULT_HOME_PAGE "index.html"
  16. #define DEFAULT_ERROR_PAGE "error.html"
  17. #define DEFAULT_INDEXOF_DIR "/downloads/"
  18. #define DEFAULT_KEEPALIVE_TIMEOUT 75000 // ms
  19. /*
  20. * @param[in] req: parsed structured http request
  21. * @param[out] resp: structured http response
  22. * @return 0: handle unfinished
  23. * http_status_code: handle done
  24. */
  25. #define HTTP_STATUS_UNFINISHED 0
  26. typedef std::function<int(HttpRequest* req, HttpResponse* resp)> http_sync_handler;
  27. typedef std::function<void(const HttpRequestPtr& req, const HttpResponseWriterPtr& writer)> http_async_handler;
  28. typedef std::function<int(const HttpContextPtr& ctx)> http_ctx_handler;
  29. struct http_handler {
  30. http_sync_handler sync_handler;
  31. http_async_handler async_handler;
  32. http_ctx_handler ctx_handler;
  33. http_handler() {}
  34. http_handler(http_sync_handler fn) : sync_handler(std::move(fn)) {}
  35. http_handler(http_async_handler fn) : async_handler(std::move(fn)) {}
  36. http_handler(http_ctx_handler fn) : ctx_handler(std::move(fn)) {}
  37. http_handler(const http_handler& rhs)
  38. : sync_handler(std::move(rhs.sync_handler))
  39. , async_handler(std::move(rhs.async_handler))
  40. , ctx_handler(std::move(rhs.ctx_handler))
  41. {}
  42. const http_handler& operator=(http_sync_handler fn) {
  43. sync_handler = std::move(fn);
  44. return *this;
  45. }
  46. const http_handler& operator=(http_async_handler fn) {
  47. async_handler = std::move(fn);
  48. return *this;
  49. }
  50. const http_handler& operator=(http_ctx_handler fn) {
  51. ctx_handler = std::move(fn);
  52. return *this;
  53. }
  54. bool isNull() {
  55. return sync_handler == NULL &&
  56. async_handler == NULL &&
  57. ctx_handler == NULL;
  58. }
  59. operator bool() {
  60. return !isNull();
  61. }
  62. };
  63. struct http_method_handler {
  64. http_method method;
  65. http_handler handler;
  66. http_method_handler() {}
  67. http_method_handler(http_method m, const http_handler& h) : method(m), handler(h) {}
  68. };
  69. // method => http_method_handler
  70. typedef std::list<http_method_handler> http_method_handlers;
  71. // path => http_method_handlers
  72. typedef std::unordered_map<std::string, std::shared_ptr<http_method_handlers>> http_api_handlers;
  73. namespace hv {
  74. struct HV_EXPORT HttpService {
  75. // preprocessor -> processor -> postprocessor
  76. http_handler preprocessor;
  77. // processor: api_handlers -> staticHandler -> errorHandler
  78. http_handler processor;
  79. http_handler postprocessor;
  80. // api service (that is http.APIServer)
  81. std::string base_url;
  82. http_api_handlers api_handlers;
  83. // file service (that is http.FileServer)
  84. http_handler staticHandler;
  85. http_handler largeFileHandler;
  86. std::string document_root;
  87. std::string home_page;
  88. std::string error_page;
  89. // indexof service (that is http.DirectoryServer)
  90. std::string index_of;
  91. http_handler errorHandler;
  92. // options
  93. int keepalive_timeout;
  94. HttpService() {
  95. // base_url = DEFAULT_BASE_URL;
  96. document_root = DEFAULT_DOCUMENT_ROOT;
  97. home_page = DEFAULT_HOME_PAGE;
  98. // error_page = DEFAULT_ERROR_PAGE;
  99. // index_of = DEFAULT_INDEXOF_DIR;
  100. keepalive_timeout = DEFAULT_KEEPALIVE_TIMEOUT;
  101. }
  102. // @retval 0 OK, else HTTP_STATUS_NOT_FOUND, HTTP_STATUS_METHOD_NOT_ALLOWED
  103. void AddApi(const char* path, http_method method, const http_handler& handler);
  104. int GetApi(const char* url, http_method method, http_handler** handler);
  105. // RESTful API /:field/ => req->query_params["field"]
  106. int GetApi(HttpRequest* req, http_handler** handler);
  107. hv::StringList Paths() {
  108. hv::StringList paths;
  109. for (auto& pair : api_handlers) {
  110. paths.emplace_back(pair.first);
  111. }
  112. return paths;
  113. }
  114. // github.com/gin-gonic/gin
  115. void Handle(const char* httpMethod, const char* relativePath, http_sync_handler handlerFunc) {
  116. AddApi(relativePath, http_method_enum(httpMethod), http_handler(handlerFunc));
  117. }
  118. void Handle(const char* httpMethod, const char* relativePath, http_async_handler handlerFunc) {
  119. AddApi(relativePath, http_method_enum(httpMethod), http_handler(handlerFunc));
  120. }
  121. void Handle(const char* httpMethod, const char* relativePath, http_ctx_handler handlerFunc) {
  122. AddApi(relativePath, http_method_enum(httpMethod), http_handler(handlerFunc));
  123. }
  124. // HEAD
  125. void HEAD(const char* relativePath, http_sync_handler handlerFunc) {
  126. Handle("HEAD", relativePath, handlerFunc);
  127. }
  128. void HEAD(const char* relativePath, http_async_handler handlerFunc) {
  129. Handle("HEAD", relativePath, handlerFunc);
  130. }
  131. void HEAD(const char* relativePath, http_ctx_handler handlerFunc) {
  132. Handle("HEAD", relativePath, handlerFunc);
  133. }
  134. // GET
  135. void GET(const char* relativePath, http_sync_handler handlerFunc) {
  136. Handle("GET", relativePath, handlerFunc);
  137. }
  138. void GET(const char* relativePath, http_async_handler handlerFunc) {
  139. Handle("GET", relativePath, handlerFunc);
  140. }
  141. void GET(const char* relativePath, http_ctx_handler handlerFunc) {
  142. Handle("GET", relativePath, handlerFunc);
  143. }
  144. // POST
  145. void POST(const char* relativePath, http_sync_handler handlerFunc) {
  146. Handle("POST", relativePath, handlerFunc);
  147. }
  148. void POST(const char* relativePath, http_async_handler handlerFunc) {
  149. Handle("POST", relativePath, handlerFunc);
  150. }
  151. void POST(const char* relativePath, http_ctx_handler handlerFunc) {
  152. Handle("POST", relativePath, handlerFunc);
  153. }
  154. // PUT
  155. void PUT(const char* relativePath, http_sync_handler handlerFunc) {
  156. Handle("PUT", relativePath, handlerFunc);
  157. }
  158. void PUT(const char* relativePath, http_async_handler handlerFunc) {
  159. Handle("PUT", relativePath, handlerFunc);
  160. }
  161. void PUT(const char* relativePath, http_ctx_handler handlerFunc) {
  162. Handle("PUT", relativePath, handlerFunc);
  163. }
  164. // DELETE
  165. // NOTE: Windows <winnt.h> #define DELETE as a macro, we have to replace DELETE with Delete.
  166. void Delete(const char* relativePath, http_sync_handler handlerFunc) {
  167. Handle("DELETE", relativePath, handlerFunc);
  168. }
  169. void Delete(const char* relativePath, http_async_handler handlerFunc) {
  170. Handle("DELETE", relativePath, handlerFunc);
  171. }
  172. void Delete(const char* relativePath, http_ctx_handler handlerFunc) {
  173. Handle("DELETE", relativePath, handlerFunc);
  174. }
  175. // PATCH
  176. void PATCH(const char* relativePath, http_sync_handler handlerFunc) {
  177. Handle("PATCH", relativePath, handlerFunc);
  178. }
  179. void PATCH(const char* relativePath, http_async_handler handlerFunc) {
  180. Handle("PATCH", relativePath, handlerFunc);
  181. }
  182. void PATCH(const char* relativePath, http_ctx_handler handlerFunc) {
  183. Handle("PATCH", relativePath, handlerFunc);
  184. }
  185. // Any
  186. void Any(const char* relativePath, http_sync_handler handlerFunc) {
  187. Handle("HEAD", relativePath, handlerFunc);
  188. Handle("GET", relativePath, handlerFunc);
  189. Handle("POST", relativePath, handlerFunc);
  190. Handle("PUT", relativePath, handlerFunc);
  191. Handle("DELETE", relativePath, handlerFunc);
  192. Handle("PATCH", relativePath, handlerFunc);
  193. }
  194. void Any(const char* relativePath, http_async_handler handlerFunc) {
  195. Handle("HEAD", relativePath, handlerFunc);
  196. Handle("GET", relativePath, handlerFunc);
  197. Handle("POST", relativePath, handlerFunc);
  198. Handle("PUT", relativePath, handlerFunc);
  199. Handle("DELETE", relativePath, handlerFunc);
  200. Handle("PATCH", relativePath, handlerFunc);
  201. }
  202. void Any(const char* relativePath, http_ctx_handler handlerFunc) {
  203. Handle("HEAD", relativePath, handlerFunc);
  204. Handle("GET", relativePath, handlerFunc);
  205. Handle("POST", relativePath, handlerFunc);
  206. Handle("PUT", relativePath, handlerFunc);
  207. Handle("DELETE", relativePath, handlerFunc);
  208. Handle("PATCH", relativePath, handlerFunc);
  209. }
  210. };
  211. }
  212. #endif // HV_HTTP_SERVICE_H_