#pragma once #define H2O_USE_LIBUV 0 extern "C" { #include "h2o.h" #include "h2o/http1.h" #include "h2o/http2.h" #include "h2o/multithread.h" } #include #include #include #include "http_data.h" #include "option.h" class ReplicationState; class HttpServer; struct h2o_custom_req_handler_t { h2o_handler_t super; HttpServer* http_server; }; struct h2o_custom_generator_t { h2o_generator_t super; h2o_custom_req_handler_t* h2o_handler; route_path *rpath; http_req* request; http_res* response; }; struct deferred_req_res_t { http_req* req; http_res* res; HttpServer* server; }; class HttpServer { private: h2o_globalconf_t config; h2o_compress_args_t compress_args; h2o_context_t ctx; h2o_accept_ctx_t* accept_ctx; h2o_hostconf_t *hostconf; h2o_socket_t* listener_socket; static const uint64_t SSL_REFRESH_INTERVAL_MS = 8 * 60 * 60 * 1000; h2o_custom_timer_t ssl_refresh_timer; http_message_dispatcher* message_dispatcher; ReplicationState* replication_state; bool exit_loop = false; std::string version; std::vector> routes; const std::string listen_address; const uint32_t listen_port; std::string ssl_cert_path; std::string ssl_cert_key_path; bool cors_enabled; bool (*auth_handler)(std::map& params, const route_path& rpath, const std::string& auth_key); static void on_accept(h2o_socket_t *listener, const char *err); int setup_ssl(const char *cert_file, const char *key_file); static void on_ssl_refresh_timeout(h2o_timer_t *entry); int create_listener(); h2o_pathconf_t *register_handler(h2o_hostconf_t *hostconf, const char *path, int (*on_req)(h2o_handler_t *, h2o_req_t *)); static std::map parse_query(const std::string& query); static int catch_all_handler(h2o_handler_t *_h2o_handler, h2o_req_t *req); static void response_proceed(h2o_generator_t *generator, h2o_req_t *req); static void on_res_generator_dispose(void *self); static int send_response(h2o_req_t *req, int status_code, const std::string & message); static int async_req_cb(void *ctx, h2o_iovec_t chunk, int is_end_stream); public: HttpServer(const std::string & version, const std::string & listen_address, uint32_t listen_port, const std::string & ssl_cert_path, const std::string & ssl_cert_key_path, bool cors_enabled); ~HttpServer(); http_message_dispatcher* get_message_dispatcher() const; ReplicationState* get_replication_state() const; bool is_alive() const; uint64_t node_state() const; void set_auth_handler(bool (*handler)(std::map& params, const route_path & rpath, const std::string & auth_key)); void get(const std::string & path, bool (*handler)(http_req & req, http_res & res), bool async_req=false, bool async_res=false); void post(const std::string & path, bool (*handler)(http_req & req, http_res & res), bool async_req=false, bool async_res=false); void put(const std::string & path, bool (*handler)(http_req & req, http_res & res), bool async_req=false, bool async_res=false); void del(const std::string & path, bool (*handler)(http_req & req, http_res & res), bool async_req=false, bool async_res=false); void on(const std::string & message, bool (*handler)(void*)); void send_message(const std::string & type, void* data); void send_response(http_req* request, http_res* response); static void stream_response(http_req& request, http_res& response); uint64_t find_route(const std::vector & path_parts, const std::string & http_method, route_path** found_rpath); bool get_route(uint64_t hash, route_path** found_rpath); int run(ReplicationState* replication_state); void stop(); void clear_timeouts(const std::vector & timers, bool trigger_callback = true); static bool on_stop_server(void *data); std::string get_version(); static constexpr const char* AUTH_HEADER = "x-typesense-api-key"; static constexpr const char* STOP_SERVER_MESSAGE = "STOP_SERVER"; static int process_request(http_req* request, http_res* response, route_path *rpath, const h2o_custom_req_handler_t *req_handler); static void on_deferred_process_request(h2o_timer_t *entry); void defer_processing(http_req& req, http_res& res, size_t timeout_ms); };