mirror of
				https://github.com/pocketpy/pocketpy
				synced 2025-10-25 22:10:17 +00:00 
			
		
		
		
	improve libhv
				
					
				
			This commit is contained in:
		
							parent
							
								
									59197d1c0c
								
							
						
					
					
						commit
						50d3c9adac
					
				| @ -12,6 +12,7 @@ set(CMAKE_CXX_STANDARD_REQUIRED ON) | |||||||
| option(BUILD_SHARED "build shared library" OFF) | option(BUILD_SHARED "build shared library" OFF) | ||||||
| option(BUILD_STATIC "build static library" ON) | option(BUILD_STATIC "build static library" ON) | ||||||
| option(BUILD_EXAMPLES "build examples" OFF) | option(BUILD_EXAMPLES "build examples" OFF) | ||||||
|  | 
 | ||||||
| option(WITH_OPENSSL "with openssl library" OFF) | option(WITH_OPENSSL "with openssl library" OFF) | ||||||
| 
 | 
 | ||||||
| add_subdirectory(${CMAKE_CURRENT_LIST_DIR}/libhv) | add_subdirectory(${CMAKE_CURRENT_LIST_DIR}/libhv) | ||||||
|  | |||||||
| @ -1,10 +1,53 @@ | |||||||
| #pragma once | #pragma once | ||||||
| 
 | 
 | ||||||
| #include "pocketpy.h" | #include "pocketpy.h" | ||||||
|  | #include "http/HttpMessage.h" | ||||||
|  | #include "base/hplatform.h" | ||||||
| 
 | 
 | ||||||
| extern "C" void pk__add_module_libhv(); | extern "C" void pk__add_module_libhv(); | ||||||
| 
 | 
 | ||||||
|  | void libhv_HttpRequest_create(py_OutRef out, HttpRequestPtr ptr); | ||||||
|  | 
 | ||||||
|  | py_Type libhv_register_HttpRequest(py_GlobalRef mod); | ||||||
| py_Type libhv_register_HttpClient(py_GlobalRef mod); | py_Type libhv_register_HttpClient(py_GlobalRef mod); | ||||||
| py_Type libhv_register_HttpServer(py_GlobalRef mod); | py_Type libhv_register_HttpServer(py_GlobalRef mod); | ||||||
| py_Type libhv_register_WebSocketClient(py_GlobalRef mod); | py_Type libhv_register_WebSocketClient(py_GlobalRef mod); | ||||||
| py_Type libhv_register_WebSocketServer(py_GlobalRef mod); | 
 | ||||||
|  | #include <deque> | ||||||
|  | #include <atomic> | ||||||
|  | 
 | ||||||
|  | template <typename T> | ||||||
|  | class libhv_MQ { | ||||||
|  | private: | ||||||
|  |     std::atomic<bool> lock; | ||||||
|  |     std::deque<T> queue; | ||||||
|  | 
 | ||||||
|  | public: | ||||||
|  |     void push(T msg) { | ||||||
|  |         while(lock.exchange(true)) { | ||||||
|  |             hv_delay(1); | ||||||
|  |         } | ||||||
|  |         queue.push_back(msg); | ||||||
|  |         lock.store(false); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     bool pop(T* msg) { | ||||||
|  |         while(lock.exchange(true)) { | ||||||
|  |             hv_delay(1); | ||||||
|  |         } | ||||||
|  |         if(queue.empty()) { | ||||||
|  |             lock.store(false); | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |         *msg = queue.front(); | ||||||
|  |         queue.pop_front(); | ||||||
|  |         lock.store(false); | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | enum class WsMessageType { | ||||||
|  |     onopen, | ||||||
|  |     onclose, | ||||||
|  |     onmessage, | ||||||
|  | }; | ||||||
|  | |||||||
| @ -1,21 +1,23 @@ | |||||||
|  | #include "HttpMessage.h" | ||||||
| #include "libhv_bindings.hpp" | #include "libhv_bindings.hpp" | ||||||
| #include "base/herr.h" | #include "base/herr.h" | ||||||
| #include "http/client/HttpClient.h" | #include "http/client/HttpClient.h" | ||||||
| 
 | 
 | ||||||
| struct libhv_HttpResponse { | struct libhv_HttpResponse { | ||||||
|     HttpResponsePtr ptr; |     HttpRequestPtr request; | ||||||
|  |     HttpResponsePtr response; | ||||||
|     bool ok; |     bool ok; | ||||||
| 
 | 
 | ||||||
|     bool is_valid() { return ok && ptr != NULL; } |     bool is_valid() { return ok && response != NULL; } | ||||||
| 
 | 
 | ||||||
|     libhv_HttpResponse() : ptr(NULL), ok(false) {} |     libhv_HttpResponse(HttpRequestPtr request) : request(request), response(NULL), ok(false) {} | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| static bool libhv_HttpResponse_status_code(int argc, py_Ref argv) { | static bool libhv_HttpResponse_status_code(int argc, py_Ref argv) { | ||||||
|     PY_CHECK_ARGC(1); |     PY_CHECK_ARGC(1); | ||||||
|     libhv_HttpResponse* resp = (libhv_HttpResponse*)py_touserdata(argv); |     libhv_HttpResponse* resp = (libhv_HttpResponse*)py_touserdata(argv); | ||||||
|     if(!resp->is_valid()) return RuntimeError("HttpResponse: no response"); |     if(!resp->is_valid()) return RuntimeError("HttpResponse: no response"); | ||||||
|     py_newint(py_retval(), resp->ptr->status_code); |     py_newint(py_retval(), resp->response->status_code); | ||||||
|     return true; |     return true; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| @ -28,7 +30,7 @@ static bool libhv_HttpResponse_headers(int argc, py_Ref argv) { | |||||||
|         py_newdict(headers); |         py_newdict(headers); | ||||||
|         py_Ref _0 = py_pushtmp(); |         py_Ref _0 = py_pushtmp(); | ||||||
|         py_Ref _1 = py_pushtmp(); |         py_Ref _1 = py_pushtmp(); | ||||||
|         for(auto& kv: resp->ptr->headers) { |         for(auto& kv: resp->response->headers) { | ||||||
|             py_newstr(_0, kv.first.c_str()); |             py_newstr(_0, kv.first.c_str()); | ||||||
|             py_newstr(_1, kv.second.c_str()); |             py_newstr(_1, kv.second.c_str()); | ||||||
|             py_dict_setitem(headers, _0, _1); |             py_dict_setitem(headers, _0, _1); | ||||||
| @ -46,8 +48,8 @@ static bool libhv_HttpResponse_text(int argc, py_Ref argv) { | |||||||
|     py_Ref text = py_getslot(argv, 1); |     py_Ref text = py_getslot(argv, 1); | ||||||
|     if(py_isnil(text)) { |     if(py_isnil(text)) { | ||||||
|         c11_sv sv; |         c11_sv sv; | ||||||
|         sv.data = resp->ptr->body.c_str(); |         sv.data = resp->response->body.c_str(); | ||||||
|         sv.size = resp->ptr->body.size(); |         sv.size = resp->response->body.size(); | ||||||
|         py_newstrv(text, sv); |         py_newstrv(text, sv); | ||||||
|     } |     } | ||||||
|     py_assign(py_retval(), text); |     py_assign(py_retval(), text); | ||||||
| @ -60,9 +62,9 @@ static bool libhv_HttpResponse_content(int argc, py_Ref argv) { | |||||||
|     if(!resp->is_valid()) return RuntimeError("HttpResponse: no response"); |     if(!resp->is_valid()) return RuntimeError("HttpResponse: no response"); | ||||||
|     py_Ref content = py_getslot(argv, 2); |     py_Ref content = py_getslot(argv, 2); | ||||||
|     if(py_isnil(content)) { |     if(py_isnil(content)) { | ||||||
|         int size = resp->ptr->body.size(); |         int size = resp->response->body.size(); | ||||||
|         unsigned char* buf = py_newbytes(content, size); |         unsigned char* buf = py_newbytes(content, size); | ||||||
|         memcpy(buf, resp->ptr->body.data(), size); |         memcpy(buf, resp->response->body.data(), size); | ||||||
|     } |     } | ||||||
|     py_assign(py_retval(), content); |     py_assign(py_retval(), content); | ||||||
|     return true; |     return true; | ||||||
| @ -72,7 +74,7 @@ static bool libhv_HttpResponse_json(int argc, py_Ref argv) { | |||||||
|     PY_CHECK_ARGC(1); |     PY_CHECK_ARGC(1); | ||||||
|     libhv_HttpResponse* resp = (libhv_HttpResponse*)py_touserdata(argv); |     libhv_HttpResponse* resp = (libhv_HttpResponse*)py_touserdata(argv); | ||||||
|     if(!resp->is_valid()) return RuntimeError("HttpResponse: no response"); |     if(!resp->is_valid()) return RuntimeError("HttpResponse: no response"); | ||||||
|     const char* source = resp->ptr->body.c_str();  // json string is null-terminated
 |     const char* source = resp->response->body.c_str();  // json string is null-terminated
 | ||||||
|     return py_json_loads(source); |     return py_json_loads(source); | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| @ -111,11 +113,19 @@ static bool libhv_HttpResponse__repr__(int argc, py_Ref argv) { | |||||||
|     if(!resp->is_valid()) { |     if(!resp->is_valid()) { | ||||||
|         py_newstr(py_retval(), "<HttpResponse: no response>"); |         py_newstr(py_retval(), "<HttpResponse: no response>"); | ||||||
|     } else { |     } else { | ||||||
|         py_newfstr(py_retval(), "<HttpResponse: %d>", (int)resp->ptr->status_code); |         py_newfstr(py_retval(), "<HttpResponse: %d>", (int)resp->response->status_code); | ||||||
|     } |     } | ||||||
|     return true; |     return true; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static bool libhv_HttpResponse_cancel(int argc, py_Ref argv) { | ||||||
|  |     PY_CHECK_ARGC(1); | ||||||
|  |     libhv_HttpResponse* resp = (libhv_HttpResponse*)py_touserdata(argv); | ||||||
|  |     resp->request->Cancel(); | ||||||
|  |     py_newnone(py_retval()); | ||||||
|  |     return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| static py_Type libhv_register_HttpResponse(py_GlobalRef mod) { | static py_Type libhv_register_HttpResponse(py_GlobalRef mod) { | ||||||
|     py_Type type = py_newtype("HttpResponse", tp_object, mod, [](void* ud) { |     py_Type type = py_newtype("HttpResponse", tp_object, mod, [](void* ud) { | ||||||
|         ((libhv_HttpResponse*)ud)->~libhv_HttpResponse(); |         ((libhv_HttpResponse*)ud)->~libhv_HttpResponse(); | ||||||
| @ -133,6 +143,8 @@ static py_Type libhv_register_HttpResponse(py_GlobalRef mod) { | |||||||
|     py_bindmagic(type, __repr__, libhv_HttpResponse__repr__); |     py_bindmagic(type, __repr__, libhv_HttpResponse__repr__); | ||||||
|     // completed
 |     // completed
 | ||||||
|     py_bindproperty(type, "completed", libhv_HttpResponse_completed, NULL); |     py_bindproperty(type, "completed", libhv_HttpResponse_completed, NULL); | ||||||
|  |     // cancel
 | ||||||
|  |     py_bindmethod(type, "cancel", libhv_HttpResponse_cancel); | ||||||
|     return type; |     return type; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -218,15 +230,15 @@ static bool libhv_HttpClient__send_request(py_Ref arg_self, | |||||||
|                                           3,  // headers, text, content
 |                                           3,  // headers, text, content
 | ||||||
|                                           sizeof(libhv_HttpResponse)); |                                           sizeof(libhv_HttpResponse)); | ||||||
|     // placement new
 |     // placement new
 | ||||||
|     new (retval) libhv_HttpResponse(); |     new (retval) libhv_HttpResponse(req); | ||||||
| 
 | 
 | ||||||
|     int code = cli->sendAsync(req, [retval](const HttpResponsePtr& resp) { |     int code = cli->sendAsync(req, [retval](const HttpResponsePtr& resp) { | ||||||
|         if(resp == NULL) { |         if(resp == NULL) { | ||||||
|             retval->ok = false; |             retval->ok = false; | ||||||
|             retval->ptr = NULL; |             retval->response = NULL; | ||||||
|         } else { |         } else { | ||||||
|             retval->ok = true; |             retval->ok = true; | ||||||
|             retval->ptr = resp; |             retval->response = resp; | ||||||
|         } |         } | ||||||
|     }); |     }); | ||||||
|     if(code != 0) { |     if(code != 0) { | ||||||
|  | |||||||
							
								
								
									
										114
									
								
								3rd/libhv/src/HttpRequest.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										114
									
								
								3rd/libhv/src/HttpRequest.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,114 @@ | |||||||
|  | #include "libhv_bindings.hpp" | ||||||
|  | #include "HttpMessage.h" | ||||||
|  | 
 | ||||||
|  | struct libhv_HttpRequest { | ||||||
|  |     HttpRequestPtr ptr; | ||||||
|  | 
 | ||||||
|  |     libhv_HttpRequest(HttpRequestPtr ptr) : ptr(ptr) {} | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | void libhv_HttpRequest_create(py_OutRef out, HttpRequestPtr ptr) { | ||||||
|  |     py_Type type = py_gettype("libhv", py_name("HttpRequest")); | ||||||
|  |     libhv_HttpRequest* self = | ||||||
|  |         (libhv_HttpRequest*)py_newobject(out, type, 2, sizeof(libhv_HttpRequest)); | ||||||
|  |     new (self) libhv_HttpRequest(ptr); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | py_Type libhv_register_HttpRequest(py_GlobalRef mod) { | ||||||
|  |     py_Type type = py_newtype("HttpRequest", tp_object, mod, [](void* ud) { | ||||||
|  |         ((libhv_HttpRequest*)ud)->~libhv_HttpRequest(); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     py_bindmagic(type, __new__, [](int argc, py_Ref argv) { | ||||||
|  |         return py_exception(tp_NotImplementedError, ""); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     py_bindproperty( | ||||||
|  |         type, | ||||||
|  |         "method", | ||||||
|  |         [](int argc, py_Ref argv) { | ||||||
|  |             PY_CHECK_ARGC(1); | ||||||
|  |             libhv_HttpRequest* req = (libhv_HttpRequest*)py_touserdata(argv); | ||||||
|  |             py_newstr(py_retval(), req->ptr->Method()); | ||||||
|  |             return true; | ||||||
|  |         }, | ||||||
|  |         NULL); | ||||||
|  | 
 | ||||||
|  |     py_bindproperty( | ||||||
|  |         type, | ||||||
|  |         "url", | ||||||
|  |         [](int argc, py_Ref argv) { | ||||||
|  |             PY_CHECK_ARGC(1); | ||||||
|  |             libhv_HttpRequest* req = (libhv_HttpRequest*)py_touserdata(argv); | ||||||
|  |             py_newstr(py_retval(), req->ptr->Url().c_str()); | ||||||
|  |             return true; | ||||||
|  |         }, | ||||||
|  |         NULL); | ||||||
|  | 
 | ||||||
|  |     py_bindproperty( | ||||||
|  |         type, | ||||||
|  |         "path", | ||||||
|  |         [](int argc, py_Ref argv) { | ||||||
|  |             PY_CHECK_ARGC(1); | ||||||
|  |             libhv_HttpRequest* req = (libhv_HttpRequest*)py_touserdata(argv); | ||||||
|  |             py_newstr(py_retval(), req->ptr->Path().c_str()); | ||||||
|  |             return true; | ||||||
|  |         }, | ||||||
|  |         NULL); | ||||||
|  | 
 | ||||||
|  |     // headers (cache in slots[0])
 | ||||||
|  |     py_bindproperty( | ||||||
|  |         type, | ||||||
|  |         "headers", | ||||||
|  |         [](int argc, py_Ref argv) { | ||||||
|  |             PY_CHECK_ARGC(1); | ||||||
|  |             libhv_HttpRequest* req = (libhv_HttpRequest*)py_touserdata(argv); | ||||||
|  |             py_Ref headers = py_getslot(argv, 0); | ||||||
|  |             if(py_isnil(headers)) { | ||||||
|  |                 py_newdict(headers); | ||||||
|  |                 py_Ref _0 = py_pushtmp(); | ||||||
|  |                 py_Ref _1 = py_pushtmp(); | ||||||
|  |                 for(auto& kv: req->ptr->headers) { | ||||||
|  |                     py_newstr(_0, kv.first.c_str());  // TODO: tolower
 | ||||||
|  |                     py_newstr(_1, kv.second.c_str()); | ||||||
|  |                     py_dict_setitem(headers, _0, _1); | ||||||
|  |                 } | ||||||
|  |                 py_shrink(2); | ||||||
|  |             } | ||||||
|  |             py_assign(py_retval(), headers); | ||||||
|  |             return true; | ||||||
|  |         }, | ||||||
|  |         NULL); | ||||||
|  | 
 | ||||||
|  |     // data (cache in slots[1])
 | ||||||
|  |     py_bindproperty( | ||||||
|  |         type, | ||||||
|  |         "data", | ||||||
|  |         [](int argc, py_Ref argv) { | ||||||
|  |             PY_CHECK_ARGC(1); | ||||||
|  |             libhv_HttpRequest* req = (libhv_HttpRequest*)py_touserdata(argv); | ||||||
|  |             py_Ref data = py_getslot(argv, 1); | ||||||
|  | 
 | ||||||
|  |             if(py_isnil(data)) { | ||||||
|  |                 auto content_type = req->ptr->ContentType(); | ||||||
|  |                 bool is_text_data = content_type == TEXT_PLAIN || | ||||||
|  |                                     content_type == APPLICATION_JSON || | ||||||
|  |                                     content_type == APPLICATION_XML || content_type == TEXT_HTML || | ||||||
|  |                                     content_type == CONTENT_TYPE_NONE; | ||||||
|  |                 if(is_text_data) { | ||||||
|  |                     c11_sv sv; | ||||||
|  |                     sv.data = req->ptr->body.data(); | ||||||
|  |                     sv.size = req->ptr->body.size(); | ||||||
|  |                     py_newstrv(data, sv); | ||||||
|  |                 } else { | ||||||
|  |                     unsigned char* buf = py_newbytes(data, req->ptr->body.size()); | ||||||
|  |                     memcpy(buf, req->ptr->body.data(), req->ptr->body.size()); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             py_assign(py_retval(), data); | ||||||
|  |             return true; | ||||||
|  |         }, | ||||||
|  |         NULL); | ||||||
|  | 
 | ||||||
|  |     return type; | ||||||
|  | } | ||||||
| @ -1,38 +1,24 @@ | |||||||
|  | #include "HttpMessage.h" | ||||||
|  | #include "WebSocketChannel.h" | ||||||
| #include "libhv_bindings.hpp" | #include "libhv_bindings.hpp" | ||||||
| #include "http/server/HttpServer.h" | #include "http/server/WebSocketServer.h" | ||||||
| #include "base/herr.h" | #include "pocketpy/pocketpy.h" | ||||||
| 
 |  | ||||||
| #include <deque> |  | ||||||
| #include <atomic> |  | ||||||
| 
 |  | ||||||
| template <typename T_in, typename T_out> |  | ||||||
| struct libhv_MQ { |  | ||||||
|     std::atomic<bool> lock_in; |  | ||||||
|     std::atomic<bool> lock_out; |  | ||||||
|     std::deque<T_in> queue_in; |  | ||||||
|     std::deque<T_out> queue_out; |  | ||||||
| 
 |  | ||||||
|     void begin_in() { |  | ||||||
|         while(lock_in.exchange(true)) { |  | ||||||
|             hv_delay(1); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void end_in() { lock_in.store(false); } |  | ||||||
| 
 |  | ||||||
|     void begin_out() { |  | ||||||
|         while(lock_out.exchange(true)) { |  | ||||||
|             hv_delay(1); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void end_out() { lock_out.store(false); } |  | ||||||
| }; |  | ||||||
| 
 | 
 | ||||||
| struct libhv_HttpServer { | struct libhv_HttpServer { | ||||||
|     hv::HttpService service; |     hv::HttpService http_service; | ||||||
|     hv::HttpServer server; |     hv::WebSocketService ws_service; | ||||||
|     libhv_MQ<HttpContextPtr, std::pair<HttpContextPtr, int>> mq; |     hv::WebSocketServer server; | ||||||
|  | 
 | ||||||
|  |     libhv_MQ<std::pair<HttpContextPtr, std::atomic<int>>*> mq; | ||||||
|  | 
 | ||||||
|  |     struct WsMessage { | ||||||
|  |         WsMessageType type; | ||||||
|  |         hv::WebSocketChannel* channel; | ||||||
|  |         HttpRequestPtr request; | ||||||
|  |         std::string body; | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     libhv_MQ<WsMessage> ws_mq; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| static bool libhv_HttpServer__new__(int argc, py_Ref argv) { | static bool libhv_HttpServer__new__(int argc, py_Ref argv) { | ||||||
| @ -49,31 +35,37 @@ static bool libhv_HttpServer__init__(int argc, py_Ref argv) { | |||||||
|     PY_CHECK_ARG_TYPE(2, tp_int); |     PY_CHECK_ARG_TYPE(2, tp_int); | ||||||
|     const char* host = py_tostr(py_arg(1)); |     const char* host = py_tostr(py_arg(1)); | ||||||
|     int port = py_toint(py_arg(2)); |     int port = py_toint(py_arg(2)); | ||||||
| 
 |  | ||||||
|     self->service.AllowCORS(); |  | ||||||
|     http_ctx_handler internal_handler = [self](const HttpContextPtr& ctx) { |  | ||||||
|         self->mq.begin_in(); |  | ||||||
|         self->mq.queue_in.push_back(ctx); |  | ||||||
|         self->mq.end_in(); |  | ||||||
| 
 |  | ||||||
|         while(true) { |  | ||||||
|             self->mq.begin_out(); |  | ||||||
|             if(!self->mq.queue_out.empty()) { |  | ||||||
|                 auto& msg = self->mq.queue_out.front(); |  | ||||||
|                 if(msg.first == ctx) { |  | ||||||
|                     self->mq.queue_out.pop_front(); |  | ||||||
|                     self->mq.end_out(); |  | ||||||
|                     return msg.second; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             self->mq.end_out(); |  | ||||||
|             hv_delay(1); |  | ||||||
|         } |  | ||||||
|     }; |  | ||||||
|     self->service.Any("*", internal_handler); |  | ||||||
|     self->server.registerHttpService(&self->service); |  | ||||||
|     self->server.setHost(host); |     self->server.setHost(host); | ||||||
|     self->server.setPort(port); |     self->server.setPort(port); | ||||||
|  | 
 | ||||||
|  |     // http
 | ||||||
|  |     self->http_service.AllowCORS(); | ||||||
|  |     http_ctx_handler internal_handler = [self](const HttpContextPtr& ctx) { | ||||||
|  |         std::pair<HttpContextPtr, std::atomic<int>> msg(ctx, 0); | ||||||
|  |         self->mq.push(&msg); | ||||||
|  |         int code; | ||||||
|  |         do { | ||||||
|  |             code = msg.second.load(); | ||||||
|  |         } while(code == 0); | ||||||
|  |         return code; | ||||||
|  |     }; | ||||||
|  |     self->http_service.Any("*", internal_handler); | ||||||
|  |     self->server.registerHttpService(&self->http_service); | ||||||
|  | 
 | ||||||
|  |     // websocket
 | ||||||
|  |     self->ws_service.onopen = [self](const WebSocketChannelPtr& channel, | ||||||
|  |                                      const HttpRequestPtr& req) { | ||||||
|  |         self->ws_mq.push({WsMessageType::onopen, channel.get(), req, ""}); | ||||||
|  |     }; | ||||||
|  |     self->ws_service.onmessage = [self](const WebSocketChannelPtr& channel, | ||||||
|  |                                         const std::string& msg) { | ||||||
|  |         self->ws_mq.push({WsMessageType::onmessage, channel.get(), nullptr, msg}); | ||||||
|  |     }; | ||||||
|  |     self->ws_service.onclose = [self](const WebSocketChannelPtr& channel) { | ||||||
|  |         self->ws_mq.push({WsMessageType::onclose, channel.get(), nullptr, ""}); | ||||||
|  |     }; | ||||||
|  |     self->server.registerWebSocketService(&self->ws_service); | ||||||
|  | 
 | ||||||
|     py_newnone(py_retval()); |     py_newnone(py_retval()); | ||||||
|     return true; |     return true; | ||||||
| } | } | ||||||
| @ -84,75 +76,45 @@ static bool libhv_HttpServer_dispatch(int argc, py_Ref argv) { | |||||||
|     py_Ref callable = py_arg(1); |     py_Ref callable = py_arg(1); | ||||||
|     if(!py_callable(callable)) return TypeError("dispatcher must be callable"); |     if(!py_callable(callable)) return TypeError("dispatcher must be callable"); | ||||||
| 
 | 
 | ||||||
|     self->mq.begin_in(); |     std::pair<HttpContextPtr, std::atomic<int>>* mq_msg; | ||||||
|     if(self->mq.queue_in.empty()) { |     if(!self->mq.pop(&mq_msg)) { | ||||||
|         self->mq.end_in(); |  | ||||||
|         py_newbool(py_retval(), false); |         py_newbool(py_retval(), false); | ||||||
|         return true; |         return true; | ||||||
|     } else { |     } else { | ||||||
|         HttpContextPtr ctx = self->mq.queue_in.front(); |         HttpContextPtr ctx = mq_msg->first; | ||||||
|         self->mq.queue_in.pop_front(); |         libhv_HttpRequest_create(py_retval(), ctx->request); | ||||||
|         self->mq.end_in(); |  | ||||||
| 
 |  | ||||||
|         const char* method = ctx->request->Method(); |  | ||||||
|         std::string path = ctx->request->Path(); |  | ||||||
|         const http_headers& headers = ctx->request->headers; |  | ||||||
|         const std::string& data = ctx->request->body; |  | ||||||
| 
 |  | ||||||
|         py_OutRef msg = py_pushtmp(); |  | ||||||
|         py_newdict(msg); |  | ||||||
|         py_Ref _0 = py_pushtmp(); |  | ||||||
|         py_Ref _1 = py_pushtmp(); |  | ||||||
|         py_Ref _2 = py_pushtmp(); |  | ||||||
|         py_Ref _3 = py_pushtmp(); |  | ||||||
| 
 |  | ||||||
|         // method
 |  | ||||||
|         py_newstr(_0, "method"); |  | ||||||
|         py_newstr(_1, method); |  | ||||||
|         py_dict_setitem(msg, _0, _1); |  | ||||||
|         // path
 |  | ||||||
|         py_newstr(_0, "path"); |  | ||||||
|         py_newstr(_1, path.c_str()); |  | ||||||
|         py_dict_setitem(msg, _0, _1); |  | ||||||
|         // headers
 |  | ||||||
|         py_newstr(_0, "headers"); |  | ||||||
|         py_newdict(_1); |  | ||||||
|         py_dict_setitem(msg, _0, _1); |  | ||||||
|         for(auto& header: headers) { |  | ||||||
|             py_newstr(_2, header.first.c_str()); |  | ||||||
|             py_newstr(_3, header.second.c_str()); |  | ||||||
|             py_dict_setitem(_1, _2, _3); |  | ||||||
|         } |  | ||||||
|         // data
 |  | ||||||
|         py_newstr(_0, "data"); |  | ||||||
|         auto content_type = ctx->request->ContentType(); |  | ||||||
|         bool is_text_data = content_type == TEXT_PLAIN || content_type == APPLICATION_JSON || |  | ||||||
|                             content_type == APPLICATION_XML || content_type == TEXT_HTML || |  | ||||||
|                             content_type == CONTENT_TYPE_NONE; |  | ||||||
|         if(is_text_data) { |  | ||||||
|             py_newstrv(_1, {data.c_str(), (int)data.size()}); |  | ||||||
|         } else { |  | ||||||
|             unsigned char* buf = py_newbytes(_1, data.size()); |  | ||||||
|             memcpy(buf, data.data(), data.size()); |  | ||||||
|         } |  | ||||||
|         py_dict_setitem(msg, _0, _1); |  | ||||||
|         py_assign(py_retval(), msg); |  | ||||||
|         py_shrink(5); |  | ||||||
| 
 |  | ||||||
|         // call dispatcher
 |         // call dispatcher
 | ||||||
|         if(!py_call(callable, 1, py_retval())) { return false; } |         if(!py_call(callable, 1, py_retval())) return false; | ||||||
| 
 | 
 | ||||||
|         py_Ref object; |         py_Ref object; | ||||||
|         int status_code = 200; |         int status_code = 200; | ||||||
|         if(py_istuple(py_retval())) { |         if(py_istuple(py_retval())) { | ||||||
|             // "Hello, world!", 200
 |             int length = py_tuple_len(py_retval()); | ||||||
|             if(py_tuple_len(py_retval()) != 2) { |             if(length == 2 || length == 3) { | ||||||
|                 return ValueError("dispatcher should return `object | tuple[object, int]`"); |                 // "Hello, world!", 200
 | ||||||
|  |                 object = py_tuple_getitem(py_retval(), 0); | ||||||
|  |                 py_ItemRef status_code_object = py_tuple_getitem(py_retval(), 1); | ||||||
|  |                 if(!py_checkint(status_code_object)) return false; | ||||||
|  |                 status_code = py_toint(status_code_object); | ||||||
|  | 
 | ||||||
|  |                 if(length == 3) { | ||||||
|  |                     // "Hello, world!", 200, {"Content-Type": "text/plain"}
 | ||||||
|  |                     py_ItemRef headers_object = py_tuple_getitem(py_retval(), 2); | ||||||
|  |                     if(!py_checktype(headers_object, tp_dict)) return false; | ||||||
|  |                     bool ok = py_dict_apply( | ||||||
|  |                         headers_object, | ||||||
|  |                         [](py_Ref key, py_Ref value, void* ctx_) { | ||||||
|  |                             if(!py_checkstr(key) || !py_checkstr(value)) return false; | ||||||
|  |                             ((hv::HttpContext*)ctx_) | ||||||
|  |                                 ->response->SetHeader(py_tostr(key), py_tostr(value)); | ||||||
|  |                             return true; | ||||||
|  |                         }, | ||||||
|  |                         ctx.get()); | ||||||
|  |                     if(!ok) return false; | ||||||
|  |                 } | ||||||
|  |             } else { | ||||||
|  |                 return TypeError("dispatcher return tuple must have 2 or 3 elements"); | ||||||
|             } |             } | ||||||
|             object = py_tuple_getitem(py_retval(), 0); |  | ||||||
|             py_ItemRef status_code_object = py_tuple_getitem(py_retval(), 1); |  | ||||||
|             if(!py_checkint(status_code_object)) return false; |  | ||||||
|             status_code = py_toint(status_code_object); |  | ||||||
|         } else { |         } else { | ||||||
|             // "Hello, world!"
 |             // "Hello, world!"
 | ||||||
|             object = py_retval(); |             object = py_retval(); | ||||||
| @ -182,9 +144,7 @@ static bool libhv_HttpServer_dispatch(int argc, py_Ref argv) { | |||||||
|             } |             } | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         self->mq.begin_out(); |         mq_msg->second.store(status_code); | ||||||
|         self->mq.queue_out.push_back({ctx, status_code}); |  | ||||||
|         self->mq.end_out(); |  | ||||||
|     } |     } | ||||||
|     py_newbool(py_retval(), true); |     py_newbool(py_retval(), true); | ||||||
|     return true; |     return true; | ||||||
| @ -194,21 +154,84 @@ static bool libhv_HttpServer_start(int argc, py_Ref argv) { | |||||||
|     PY_CHECK_ARGC(1); |     PY_CHECK_ARGC(1); | ||||||
|     libhv_HttpServer* self = (libhv_HttpServer*)py_touserdata(py_arg(0)); |     libhv_HttpServer* self = (libhv_HttpServer*)py_touserdata(py_arg(0)); | ||||||
|     int code = self->server.start(); |     int code = self->server.start(); | ||||||
|     if(code != 0) { |     py_newint(py_retval(), code); | ||||||
|         return RuntimeError("HttpServer start failed: %s (%d)", hv_strerror(code), code); |  | ||||||
|     } |  | ||||||
|     py_newnone(py_retval()); |  | ||||||
|     return true; |     return true; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static bool libhv_HttpServer_stop(int argc, py_Ref argv) { | static bool libhv_HttpServer_stop(int argc, py_Ref argv) { | ||||||
|     PY_CHECK_ARGC(1); |     PY_CHECK_ARGC(1); | ||||||
|     libhv_HttpServer* self = (libhv_HttpServer*)py_touserdata(py_arg(0)); |     libhv_HttpServer* self = (libhv_HttpServer*)py_touserdata(py_arg(0)); | ||||||
|     self->server.stop(); |     int code = self->server.stop(); | ||||||
|  |     py_newint(py_retval(), code); | ||||||
|  |     return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static bool libhv_HttpServer_ws_set_ping_interval(int argc, py_Ref argv) { | ||||||
|  |     PY_CHECK_ARGC(2); | ||||||
|  |     libhv_HttpServer* self = (libhv_HttpServer*)py_touserdata(py_arg(0)); | ||||||
|  |     PY_CHECK_ARG_TYPE(1, tp_int); | ||||||
|  |     int interval = py_toint(py_arg(1)); | ||||||
|  |     self->ws_service.setPingInterval(interval); | ||||||
|     py_newnone(py_retval()); |     py_newnone(py_retval()); | ||||||
|     return true; |     return true; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static bool libhv_HttpServer_ws_send(int argc, py_Ref argv) { | ||||||
|  |     PY_CHECK_ARGC(3); | ||||||
|  |     libhv_HttpServer* self = (libhv_HttpServer*)py_touserdata(py_arg(0)); | ||||||
|  |     PY_CHECK_ARG_TYPE(1, tp_int); | ||||||
|  |     PY_CHECK_ARG_TYPE(2, tp_str); | ||||||
|  |     py_i64 channel = py_toint(py_arg(1)); | ||||||
|  |     const char* msg = py_tostr(py_arg(2)); | ||||||
|  | 
 | ||||||
|  |     hv::WebSocketChannel* p_channel = reinterpret_cast<hv::WebSocketChannel*>(channel); | ||||||
|  |     int code = p_channel->send(msg); | ||||||
|  |     py_newint(py_retval(), code); | ||||||
|  |     return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static bool libhv_HttpServer_ws_recv(int argc, py_Ref argv) { | ||||||
|  |     PY_CHECK_ARGC(1); | ||||||
|  |     libhv_HttpServer* self = (libhv_HttpServer*)py_touserdata(py_arg(0)); | ||||||
|  |     libhv_HttpServer::WsMessage msg; | ||||||
|  |     if(!self->ws_mq.pop(&msg)) { | ||||||
|  |         py_newnone(py_retval()); | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  |     py_newtuple(py_retval(), 2); | ||||||
|  |     switch(msg.type) { | ||||||
|  |         case WsMessageType::onopen: { | ||||||
|  |             // "onopen", (channel, request)
 | ||||||
|  |             assert(msg.request != nullptr); | ||||||
|  |             py_newstr(py_tuple_getitem(py_retval(), 0), "onopen"); | ||||||
|  |             py_Ref args = py_tuple_getitem(py_retval(), 1); | ||||||
|  |             py_newtuple(args, 2); | ||||||
|  |             py_newint(py_tuple_getitem(args, 0), (py_i64)msg.channel); | ||||||
|  |             libhv_HttpRequest_create(py_tuple_getitem(args, 1), msg.request); | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |         case WsMessageType::onclose: { | ||||||
|  |             // "onclose", channel
 | ||||||
|  |             py_newstr(py_tuple_getitem(py_retval(), 0), "onclose"); | ||||||
|  |             py_newint(py_tuple_getitem(py_retval(), 1), (py_i64)msg.channel); | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |         case WsMessageType::onmessage: { | ||||||
|  |             // "onmessage", (channel, body)
 | ||||||
|  |             py_newstr(py_tuple_getitem(py_retval(), 0), "onmessage"); | ||||||
|  |             py_Ref args = py_tuple_getitem(py_retval(), 1); | ||||||
|  |             py_newtuple(args, 2); | ||||||
|  |             py_newint(py_tuple_getitem(args, 0), (py_i64)msg.channel); | ||||||
|  |             c11_sv sv; | ||||||
|  |             sv.data = msg.body.data(); | ||||||
|  |             sv.size = msg.body.size(); | ||||||
|  |             py_newstrv(py_tuple_getitem(args, 1), sv); | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| py_Type libhv_register_HttpServer(py_GlobalRef mod) { | py_Type libhv_register_HttpServer(py_GlobalRef mod) { | ||||||
|     py_Type type = py_newtype("HttpServer", tp_object, mod, [](void* ud) { |     py_Type type = py_newtype("HttpServer", tp_object, mod, [](void* ud) { | ||||||
|         libhv_HttpServer* self = (libhv_HttpServer*)ud; |         libhv_HttpServer* self = (libhv_HttpServer*)ud; | ||||||
| @ -217,8 +240,12 @@ py_Type libhv_register_HttpServer(py_GlobalRef mod) { | |||||||
| 
 | 
 | ||||||
|     py_bindmagic(type, __new__, libhv_HttpServer__new__); |     py_bindmagic(type, __new__, libhv_HttpServer__new__); | ||||||
|     py_bindmagic(type, __init__, libhv_HttpServer__init__); |     py_bindmagic(type, __init__, libhv_HttpServer__init__); | ||||||
|     py_bindmethod(type, "dispatch", libhv_HttpServer_dispatch); |  | ||||||
|     py_bindmethod(type, "start", libhv_HttpServer_start); |     py_bindmethod(type, "start", libhv_HttpServer_start); | ||||||
|     py_bindmethod(type, "stop", libhv_HttpServer_stop); |     py_bindmethod(type, "stop", libhv_HttpServer_stop); | ||||||
|  |     py_bindmethod(type, "dispatch", libhv_HttpServer_dispatch); | ||||||
|  | 
 | ||||||
|  |     py_bindmethod(type, "ws_set_ping_interval", libhv_HttpServer_ws_set_ping_interval); | ||||||
|  |     py_bindmethod(type, "ws_send", libhv_HttpServer_ws_send); | ||||||
|  |     py_bindmethod(type, "ws_recv", libhv_HttpServer_ws_recv); | ||||||
|     return type; |     return type; | ||||||
| } | } | ||||||
| @ -1,7 +1,125 @@ | |||||||
|  | #include "HttpMessage.h" | ||||||
| #include "libhv_bindings.hpp" | #include "libhv_bindings.hpp" | ||||||
|  | #include "pocketpy/pocketpy.h" | ||||||
| #include "http/client/WebSocketClient.h" | #include "http/client/WebSocketClient.h" | ||||||
| 
 | 
 | ||||||
|  | struct libhv_WebSocketClient { | ||||||
|  |     hv::WebSocketClient ws; | ||||||
|  | 
 | ||||||
|  |     libhv_MQ<std::pair<WsMessageType, std::string>> mq; | ||||||
|  | 
 | ||||||
|  |     libhv_WebSocketClient() { | ||||||
|  |         ws.onopen = [this]() { | ||||||
|  |             mq.push({WsMessageType::onopen, ""}); | ||||||
|  |         }; | ||||||
|  |         ws.onclose = [this]() { | ||||||
|  |             mq.push({WsMessageType::onclose, ""}); | ||||||
|  |         }; | ||||||
|  |         ws.onmessage = [this](const std::string& msg) { | ||||||
|  |             mq.push({WsMessageType::onmessage, msg}); | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         // reconnect: 1,2,4,8,10,10,10...
 | ||||||
|  |         reconn_setting_t reconn; | ||||||
|  |         reconn_setting_init(&reconn); | ||||||
|  |         reconn.min_delay = 1000; | ||||||
|  |         reconn.max_delay = 10000; | ||||||
|  |         reconn.delay_policy = 2; | ||||||
|  |         ws.setReconnect(&reconn); | ||||||
|  |     } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
| py_Type libhv_register_WebSocketClient(py_GlobalRef mod) { | py_Type libhv_register_WebSocketClient(py_GlobalRef mod) { | ||||||
|     py_Type type = py_newtype("WebSocketClient", tp_object, mod, NULL); |     py_Type type = py_newtype("WebSocketClient", tp_object, mod, [](void* ud) { | ||||||
|  |         libhv_WebSocketClient* self = (libhv_WebSocketClient*)ud; | ||||||
|  |         self->~libhv_WebSocketClient(); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     py_bindmagic(type, __new__, [](int argc, py_Ref argv) { | ||||||
|  |         PY_CHECK_ARGC(1); | ||||||
|  |         libhv_WebSocketClient* self = (libhv_WebSocketClient*) | ||||||
|  |             py_newobject(py_retval(), py_totype(argv), 0, sizeof(libhv_WebSocketClient)); | ||||||
|  |         new (self) libhv_WebSocketClient(); | ||||||
|  |         return true; | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     py_bindmethod(type, "open", [](int argc, py_Ref argv) { | ||||||
|  |         libhv_WebSocketClient* self = (libhv_WebSocketClient*)py_touserdata(argv); | ||||||
|  |         PY_CHECK_ARG_TYPE(1, tp_str); | ||||||
|  |         const char* url = py_tostr(py_arg(1)); | ||||||
|  |         http_headers headers = DefaultHeaders; | ||||||
|  |         if(argc == 2) { | ||||||
|  |             // open(self, url)
 | ||||||
|  |         } else if(argc == 3) { | ||||||
|  |             // open(self, url, headers)
 | ||||||
|  |             if(!py_checktype(py_arg(2), tp_dict)) return false; | ||||||
|  |             bool ok = py_dict_apply( | ||||||
|  |                 py_arg(2), | ||||||
|  |                 [](py_Ref key, py_Ref value, void* ctx) { | ||||||
|  |                     http_headers* p_headers = (http_headers*)ctx; | ||||||
|  |                     if(!py_checkstr(key)) return false; | ||||||
|  |                     if(!py_checkstr(value)) return false; | ||||||
|  |                     p_headers->operator[](py_tostr(key)) = py_tostr(value); | ||||||
|  |                     return true; | ||||||
|  |                 }, | ||||||
|  |                 &headers); | ||||||
|  |             if(!ok) return false; | ||||||
|  |         } else { | ||||||
|  |             return TypeError("open() takes 2 or 3 arguments"); | ||||||
|  |         } | ||||||
|  |         int code = self->ws.open(url, headers); | ||||||
|  |         py_newint(py_retval(), code); | ||||||
|  |         return true; | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     py_bindmethod(type, "close", [](int argc, py_Ref argv) { | ||||||
|  |         PY_CHECK_ARGC(1); | ||||||
|  |         libhv_WebSocketClient* self = (libhv_WebSocketClient*)py_touserdata(argv); | ||||||
|  |         int code = self->ws.close(); | ||||||
|  |         py_newint(py_retval(), code); | ||||||
|  |         return true; | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     py_bindmethod(type, "send", [](int argc, py_Ref argv) { | ||||||
|  |         PY_CHECK_ARGC(2); | ||||||
|  |         libhv_WebSocketClient* self = (libhv_WebSocketClient*)py_touserdata(argv); | ||||||
|  |         PY_CHECK_ARG_TYPE(1, tp_str); | ||||||
|  |         const char* msg = py_tostr(py_arg(1)); | ||||||
|  |         int code = self->ws.send(msg); | ||||||
|  |         py_newint(py_retval(), code); | ||||||
|  |         return true; | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     py_bindmethod(type, "recv", [](int argc, py_Ref argv) { | ||||||
|  |         PY_CHECK_ARGC(1); | ||||||
|  |         libhv_WebSocketClient* self = (libhv_WebSocketClient*)py_touserdata(py_arg(0)); | ||||||
|  | 
 | ||||||
|  |         std::pair<WsMessageType, std::string> mq_msg; | ||||||
|  |         if(!self->mq.pop(&mq_msg)) { | ||||||
|  |             py_newnone(py_retval()); | ||||||
|  |             return true; | ||||||
|  |         } else { | ||||||
|  |             py_newtuple(py_retval(), 2); | ||||||
|  |             switch(mq_msg.first) { | ||||||
|  |                 case WsMessageType::onopen: { | ||||||
|  |                     py_newstr(py_tuple_getitem(py_retval(), 0), "onopen"); | ||||||
|  |                     py_newnone(py_tuple_getitem(py_retval(), 1)); | ||||||
|  |                     break; | ||||||
|  |                 } | ||||||
|  |                 case WsMessageType::onclose: { | ||||||
|  |                     py_newstr(py_tuple_getitem(py_retval(), 0), "onclose"); | ||||||
|  |                     py_newnone(py_tuple_getitem(py_retval(), 1)); | ||||||
|  |                     break; | ||||||
|  |                 } | ||||||
|  |                 case WsMessageType::onmessage: { | ||||||
|  |                     py_newstr(py_tuple_getitem(py_retval(), 0), "onmessage"); | ||||||
|  |                     py_newstrv(py_tuple_getitem(py_retval(), 1), | ||||||
|  |                                {mq_msg.second.data(), (int)mq_msg.second.size()}); | ||||||
|  |                     break; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|     return type; |     return type; | ||||||
| } | } | ||||||
| @ -1,7 +0,0 @@ | |||||||
| #include "libhv_bindings.hpp" |  | ||||||
| #include "http/server/WebSocketServer.h" |  | ||||||
| 
 |  | ||||||
| py_Type libhv_register_WebSocketServer(py_GlobalRef mod) { |  | ||||||
|     py_Type type = py_newtype("WebSocketServer", tp_object, mod, NULL); |  | ||||||
|     return type; |  | ||||||
| } |  | ||||||
| @ -1,10 +1,20 @@ | |||||||
| #include "libhv_bindings.hpp" | #include "libhv_bindings.hpp" | ||||||
|  | #include "base/herr.h" | ||||||
| 
 | 
 | ||||||
| extern "C" void pk__add_module_libhv() { | extern "C" void pk__add_module_libhv() { | ||||||
|     py_GlobalRef mod = py_newmodule("libhv"); |     py_GlobalRef mod = py_newmodule("libhv"); | ||||||
| 
 | 
 | ||||||
|  |     libhv_register_HttpRequest(mod); | ||||||
|     libhv_register_HttpClient(mod); |     libhv_register_HttpClient(mod); | ||||||
|     libhv_register_HttpServer(mod); |     libhv_register_HttpServer(mod); | ||||||
|     libhv_register_WebSocketClient(mod); |     libhv_register_WebSocketClient(mod); | ||||||
|     libhv_register_WebSocketServer(mod); | 
 | ||||||
|  |     py_bindfunc(mod, "strerror", [](int argc, py_Ref argv) { | ||||||
|  |         PY_CHECK_ARGC(1); | ||||||
|  |         PY_CHECK_ARG_TYPE(0, tp_int); | ||||||
|  |         int code = py_toint(py_arg(0)); | ||||||
|  |         const char* msg = hv_strerror(code); | ||||||
|  |         py_newstr(py_retval(), msg); | ||||||
|  |         return true; | ||||||
|  |     }); | ||||||
| } | } | ||||||
|  | |||||||
| @ -7,11 +7,17 @@ label: libhv | |||||||
| This module is optional. Set option `PK_BUILD_MODULE_LIBHV` to `ON` in your `CMakeLists.txt` to enable it. | This module is optional. Set option `PK_BUILD_MODULE_LIBHV` to `ON` in your `CMakeLists.txt` to enable it. | ||||||
| !!! | !!! | ||||||
| 
 | 
 | ||||||
|  | `libhv` is a git submodule located at `3rd/libhv/libhv`. If you cannot find it, please run the following command to initialize the submodule: | ||||||
|  | 
 | ||||||
|  | ```bash | ||||||
|  | git submodule update --init --recursive | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
| Simple bindings for [libhv](https://github.com/ithewei/libhv), which provides cross platform implementation of the following: | Simple bindings for [libhv](https://github.com/ithewei/libhv), which provides cross platform implementation of the following: | ||||||
| + HTTP server | + HTTP server | ||||||
| + HTTP client | + HTTP client | ||||||
| + WebSocket server (TODO) | + WebSocket server | ||||||
| + WebSocket client (TODO) | + WebSocket client | ||||||
| 
 | 
 | ||||||
| #### Source code | #### Source code | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -1,7 +1,15 @@ | |||||||
| from typing import Literal, Generator, Callable | from typing import Literal, Generator, Callable | ||||||
| 
 | 
 | ||||||
|  | WsMessageType = Literal['onopen', 'onclose', 'onmessage'] | ||||||
|  | WsChannelId = int | ||||||
|  | HttpStatusCode = int | ||||||
|  | HttpHeaders = dict[str, str] | ||||||
|  | ErrorCode = int | ||||||
|  | 
 | ||||||
| class Future[T]: | class Future[T]: | ||||||
|  |     @property | ||||||
|     def completed(self) -> bool: ... |     def completed(self) -> bool: ... | ||||||
|  |     def cancel(self) -> None: ... | ||||||
|     def __iter__(self) -> Generator[T, None, None]: ... |     def __iter__(self) -> Generator[T, None, None]: ... | ||||||
| 
 | 
 | ||||||
| class HttpResponse(Future['HttpResponse']): | class HttpResponse(Future['HttpResponse']): | ||||||
| @ -18,21 +26,75 @@ class HttpResponse(Future['HttpResponse']): | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| class HttpClient: | class HttpClient: | ||||||
|     def get(self, url: str, params=None, headers=None, timeout=10) -> HttpResponse: ... |     def get(self, url: str, /, params=None, headers=None, timeout=10) -> HttpResponse: ... | ||||||
|     def post(self, url: str, params=None, headers=None, data=None, json=None, timeout=10) -> HttpResponse: ... |     def post(self, url: str, /, params=None, headers=None, data=None, json=None, timeout=10) -> HttpResponse: ... | ||||||
|     def put(self, url: str, params=None, headers=None, data=None, json=None, timeout=10) -> HttpResponse: ... |     def put(self, url: str, /, params=None, headers=None, data=None, json=None, timeout=10) -> HttpResponse: ... | ||||||
|     def delete(self, url: str, params=None, headers=None, timeout=10) -> HttpResponse: ... |     def delete(self, url: str, /, params=None, headers=None, timeout=10) -> HttpResponse: ... | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | class HttpRequest: | ||||||
|  |     @property | ||||||
|  |     def method(self) -> Literal['GET', 'POST', 'PUT', 'DELETE']: ... | ||||||
|  |     @property | ||||||
|  |     def path(self) -> str: ... | ||||||
|  |     @property | ||||||
|  |     def url(self) -> str: ... | ||||||
|  |     @property | ||||||
|  |     def headers(self) -> HttpHeaders: ... | ||||||
|  |     @property | ||||||
|  |     def data(self) -> str | bytes: ... | ||||||
|  | 
 | ||||||
| class HttpServer: | class HttpServer: | ||||||
|     def __init__(self, host: str, port: int) -> None: ... |     def __init__(self, host: str, port: int, /) -> None: ... | ||||||
|     def dispatch(self, fn: Callable[[dict], object | tuple[object, int]]) -> bool: ... |     def start(self) -> ErrorCode: ... | ||||||
|     def start(self) -> None: ... |     def stop(self) -> ErrorCode: ... | ||||||
|     def stop(self) -> None: ... |     def dispatch[T](self, fn: Callable[ | ||||||
|  |         [HttpRequest], | ||||||
|  |         T | tuple[T, HttpStatusCode] | tuple[T, HttpStatusCode, HttpHeaders] | ||||||
|  |         ], /) -> bool: | ||||||
|  |         """Dispatch one HTTP request through `fn`. `fn` should return one of the following: | ||||||
| 
 | 
 | ||||||
|  |         + object | ||||||
|  |         + (object, status_code) | ||||||
|  |         + (object, status_code, headers) | ||||||
|  |          | ||||||
|  |         Return `True` if dispatched, otherwise `False`. | ||||||
|  |         """ | ||||||
|  | 
 | ||||||
|  |     def ws_set_ping_interval(self, milliseconds: int, /) -> None: | ||||||
|  |         """Set WebSocket ping interval in milliseconds.""" | ||||||
|  | 
 | ||||||
|  |     def ws_send(self, channel: WsChannelId, data: str, /) -> ErrorCode: | ||||||
|  |         """Send WebSocket message through `channel`.""" | ||||||
|  | 
 | ||||||
|  |     def ws_recv(self) -> tuple[ | ||||||
|  |         WsMessageType, | ||||||
|  |         tuple[WsChannelId, HttpRequest] | WsChannelId | tuple[WsChannelId, str] | ||||||
|  |         ] | None: | ||||||
|  |         """Receive one WebSocket message. | ||||||
|  |         Return one of the following or `None` if nothing to receive. | ||||||
|  |          | ||||||
|  |         + `"onopen"`: (channel, request) | ||||||
|  |         + `"onclose"`: channel | ||||||
|  |         + `"onmessage"`: (channel, body) | ||||||
|  |         """ | ||||||
| 
 | 
 | ||||||
| class WebSocketClient: | class WebSocketClient: | ||||||
|     pass |     def open(self, url: str, headers=None, /) -> ErrorCode: ... | ||||||
|  |     def close(self) -> ErrorCode: ... | ||||||
| 
 | 
 | ||||||
| class WebSocketServer: |     def send(self, data: str, /) -> ErrorCode: | ||||||
|     pass |         """Send WebSocket message.""" | ||||||
|  | 
 | ||||||
|  |     def recv(self) -> tuple[WsMessageType, str | None] | None: | ||||||
|  |         """Receive one WebSocket message. | ||||||
|  |         Return one of the following or `None` if nothing to receive. | ||||||
|  |          | ||||||
|  |         + `"onopen"`: `None` | ||||||
|  |         + `"onclose"`: `None` | ||||||
|  |         + `"onmessage"`: body | ||||||
|  |         """ | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def strerror(errno: ErrorCode, /) -> str: | ||||||
|  |     """Get error message by errno via `hv_strerror`.""" | ||||||
|  | |||||||
| @ -152,7 +152,7 @@ bool py_callcfunc(py_CFunction f, int argc, py_Ref argv) { | |||||||
|     } |     } | ||||||
|     if(py_checkexc(true)) { |     if(py_checkexc(true)) { | ||||||
|         const char* name = py_tpname(pk_current_vm->curr_exception.type); |         const char* name = py_tpname(pk_current_vm->curr_exception.type); | ||||||
|         c11__abort("py_CFunction returns `true`, but `%s` is set!", name); |         c11__abort("py_CFunction returns `true`, but `%s` was set!", name); | ||||||
|     } |     } | ||||||
|     return true; |     return true; | ||||||
| } | } | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user