| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278 |
- #ifndef HV_HTTP_SERVICE_H_
- #define HV_HTTP_SERVICE_H_
- #include <string>
- #include <map>
- #include <unordered_map>
- #include <vector>
- #include <list>
- #include <memory>
- #include <functional>
- #include "hexport.h"
- #include "HttpMessage.h"
- #include "HttpResponseWriter.h"
- #include "HttpContext.h"
- #define DEFAULT_BASE_URL "/api/v1"
- #define DEFAULT_DOCUMENT_ROOT "/var/www/html"
- #define DEFAULT_HOME_PAGE "index.html"
- #define DEFAULT_ERROR_PAGE "error.html"
- #define DEFAULT_INDEXOF_DIR "/downloads/"
- #define DEFAULT_KEEPALIVE_TIMEOUT 75000 // ms
- // for FileCache
- #define MAX_FILE_CACHE_SIZE (1 << 22) // 4M
- #define DEFAULT_FILE_CACHE_STAT_INTERVAL 10 // s
- #define DEFAULT_FILE_CACHE_EXPIRED_TIME 60 // s
- /*
- * @param[in] req: parsed structured http request
- * @param[out] resp: structured http response
- * @return 0: handle next
- * http_status_code: handle done
- */
- #define HTTP_STATUS_NEXT 0
- #define HTTP_STATUS_UNFINISHED 0
- // NOTE: http_sync_handler run on IO thread
- typedef std::function<int(HttpRequest* req, HttpResponse* resp)> http_sync_handler;
- // NOTE: http_async_handler run on hv::async threadpool
- typedef std::function<void(const HttpRequestPtr& req, const HttpResponseWriterPtr& writer)> http_async_handler;
- // NOTE: http_ctx_handler run on IO thread, you can easily post HttpContextPtr to your consumer thread for processing.
- typedef std::function<int(const HttpContextPtr& ctx)> http_ctx_handler;
- // NOTE: http_state_handler run on IO thread
- typedef std::function<int(const HttpContextPtr& ctx, http_parser_state state, const char* data, size_t size)> http_state_handler;
- struct http_handler {
- http_sync_handler sync_handler;
- http_async_handler async_handler;
- http_ctx_handler ctx_handler;
- http_state_handler state_handler;
- http_handler() {}
- http_handler(http_sync_handler fn) : sync_handler(std::move(fn)) {}
- http_handler(http_async_handler fn) : async_handler(std::move(fn)) {}
- http_handler(http_ctx_handler fn) : ctx_handler(std::move(fn)) {}
- http_handler(http_state_handler fn) : state_handler(std::move(fn)) {}
- http_handler(const http_handler& rhs)
- : sync_handler(std::move(rhs.sync_handler))
- , async_handler(std::move(rhs.async_handler))
- , ctx_handler(std::move(rhs.ctx_handler))
- , state_handler(std::move(rhs.state_handler))
- {}
- const http_handler& operator=(http_sync_handler fn) {
- sync_handler = std::move(fn);
- return *this;
- }
- const http_handler& operator=(http_async_handler fn) {
- async_handler = std::move(fn);
- return *this;
- }
- const http_handler& operator=(http_ctx_handler fn) {
- ctx_handler = std::move(fn);
- return *this;
- }
- const http_handler& operator=(http_state_handler fn) {
- state_handler = std::move(fn);
- return *this;
- }
- bool isNull() {
- return sync_handler == NULL &&
- async_handler == NULL &&
- ctx_handler == NULL;
- }
- operator bool() {
- return !isNull();
- }
- };
- typedef std::vector<http_handler> http_handlers;
- struct http_method_handler {
- http_method method;
- http_handler handler;
- http_method_handler() {}
- http_method_handler(http_method m, const http_handler& h) : method(m), handler(h) {}
- };
- // method => http_method_handler
- typedef std::list<http_method_handler> http_method_handlers;
- // path => http_method_handlers
- typedef std::unordered_map<std::string, std::shared_ptr<http_method_handlers>> http_path_handlers;
- namespace hv {
- struct HV_EXPORT HttpService {
- // preprocessor -> middleware -> processor -> postprocessor
- http_handler preprocessor;
- http_handlers middleware;
- // processor: pathHandlers -> staticHandler -> errorHandler
- http_handler processor;
- http_handler postprocessor;
- // api service (that is http.ApiServer)
- std::string base_url;
- http_path_handlers pathHandlers;
- // file service (that is http.FileServer)
- http_handler staticHandler;
- http_handler largeFileHandler;
- std::string document_root;
- std::string home_page;
- std::string error_page;
- // nginx: location => root
- std::map<std::string, std::string, std::greater<std::string>> staticDirs;
- // indexof service (that is http.DirectoryServer)
- std::string index_of;
- http_handler errorHandler;
- // proxy service (that is http.ProxyServer)
- // nginx: location => proxy_pass
- std::map<std::string, std::string, std::greater<std::string>> proxies;
- StringList trustProxies;
- StringList noProxies;
- int proxy_connect_timeout;
- int proxy_read_timeout;
- int proxy_write_timeout;
- // options
- int keepalive_timeout;
- int max_file_cache_size; // cache small file
- int file_cache_stat_interval; // stat file is modified
- int file_cache_expired_time; // remove expired file cache
- /*
- * @test limit_rate
- * @build make examples
- * @server bin/httpd -c etc/httpd.conf -s restart -d
- * @client bin/wget http://127.0.0.1:8080/downloads/test.zip
- */
- int limit_rate; // limit send rate, unit: KB/s
- unsigned enable_access_log :1;
- unsigned enable_forward_proxy :1;
- HttpService() {
- // base_url = DEFAULT_BASE_URL;
- document_root = DEFAULT_DOCUMENT_ROOT;
- home_page = DEFAULT_HOME_PAGE;
- // error_page = DEFAULT_ERROR_PAGE;
- // index_of = DEFAULT_INDEXOF_DIR;
- proxy_connect_timeout = DEFAULT_CONNECT_TIMEOUT;
- proxy_read_timeout = 0;
- proxy_write_timeout = 0;
- keepalive_timeout = DEFAULT_KEEPALIVE_TIMEOUT;
- max_file_cache_size = MAX_FILE_CACHE_SIZE;
- file_cache_stat_interval = DEFAULT_FILE_CACHE_STAT_INTERVAL;
- file_cache_expired_time = DEFAULT_FILE_CACHE_EXPIRED_TIME;
- limit_rate = -1; // unlimited
- enable_access_log = 1;
- enable_forward_proxy = 0;
- }
- void AddRoute(const char* path, http_method method, const http_handler& handler);
- // @retval 0 OK, else HTTP_STATUS_NOT_FOUND, HTTP_STATUS_METHOD_NOT_ALLOWED
- int GetRoute(const char* url, http_method method, http_handler** handler);
- // RESTful API /:field/ => req->query_params["field"]
- int GetRoute(HttpRequest* req, http_handler** handler);
- // Static("/", "/var/www/html")
- void Static(const char* path, const char* dir);
- // @retval / => /var/www/html/index.html
- std::string GetStaticFilepath(const char* path);
- // https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS
- void AllowCORS();
- // proxy
- void AddTrustProxy(const char* host);
- void AddNoProxy(const char* host);
- bool IsTrustProxy(const char* host);
- // forward proxy
- void EnableForwardProxy() { enable_forward_proxy = 1; }
- // reverse proxy
- // Proxy("/api/v1/", "http://www.httpbin.org/");
- void Proxy(const char* path, const char* url);
- // @retval /api/v1/test => http://www.httpbin.org/test
- std::string GetProxyUrl(const char* path);
- hv::StringList Paths() {
- hv::StringList paths;
- for (auto& pair : pathHandlers) {
- paths.emplace_back(pair.first);
- }
- return paths;
- }
- // Handler = [ http_sync_handler, http_ctx_handler ]
- template<typename Handler>
- void Use(Handler handlerFunc) {
- middleware.emplace_back(handlerFunc);
- }
- // Inspired by github.com/gin-gonic/gin
- // Handler = [ http_sync_handler, http_async_handler, http_ctx_handler, http_state_handler ]
- template<typename Handler>
- void Handle(const char* httpMethod, const char* relativePath, Handler handlerFunc) {
- AddRoute(relativePath, http_method_enum(httpMethod), http_handler(handlerFunc));
- }
- // HEAD
- template<typename Handler>
- void HEAD(const char* relativePath, Handler handlerFunc) {
- Handle("HEAD", relativePath, handlerFunc);
- }
- // GET
- template<typename Handler>
- void GET(const char* relativePath, Handler handlerFunc) {
- Handle("GET", relativePath, handlerFunc);
- }
- // POST
- template<typename Handler>
- void POST(const char* relativePath, Handler handlerFunc) {
- Handle("POST", relativePath, handlerFunc);
- }
- // PUT
- template<typename Handler>
- void PUT(const char* relativePath, Handler handlerFunc) {
- Handle("PUT", relativePath, handlerFunc);
- }
- // DELETE
- // NOTE: Windows <winnt.h> #define DELETE as a macro, we have to replace DELETE with Delete.
- template<typename Handler>
- void Delete(const char* relativePath, Handler handlerFunc) {
- Handle("DELETE", relativePath, handlerFunc);
- }
- // PATCH
- template<typename Handler>
- void PATCH(const char* relativePath, Handler handlerFunc) {
- Handle("PATCH", relativePath, handlerFunc);
- }
- // Any
- template<typename Handler>
- void Any(const char* relativePath, Handler handlerFunc) {
- Handle("HEAD", relativePath, handlerFunc);
- Handle("GET", relativePath, handlerFunc);
- Handle("POST", relativePath, handlerFunc);
- Handle("PUT", relativePath, handlerFunc);
- Handle("DELETE", relativePath, handlerFunc);
- Handle("PATCH", relativePath, handlerFunc);
- }
- };
- }
- #endif // HV_HTTP_SERVICE_H_
|