| 1 | /**************************************************************************/ | 
|---|
| 2 | /*  wsl_peer.h                                                            */ | 
|---|
| 3 | /**************************************************************************/ | 
|---|
| 4 | /*                         This file is part of:                          */ | 
|---|
| 5 | /*                             GODOT ENGINE                               */ | 
|---|
| 6 | /*                        https://godotengine.org                         */ | 
|---|
| 7 | /**************************************************************************/ | 
|---|
| 8 | /* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */ | 
|---|
| 9 | /* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur.                  */ | 
|---|
| 10 | /*                                                                        */ | 
|---|
| 11 | /* Permission is hereby granted, free of charge, to any person obtaining  */ | 
|---|
| 12 | /* a copy of this software and associated documentation files (the        */ | 
|---|
| 13 | /* "Software"), to deal in the Software without restriction, including    */ | 
|---|
| 14 | /* without limitation the rights to use, copy, modify, merge, publish,    */ | 
|---|
| 15 | /* distribute, sublicense, and/or sell copies of the Software, and to     */ | 
|---|
| 16 | /* permit persons to whom the Software is furnished to do so, subject to  */ | 
|---|
| 17 | /* the following conditions:                                              */ | 
|---|
| 18 | /*                                                                        */ | 
|---|
| 19 | /* The above copyright notice and this permission notice shall be         */ | 
|---|
| 20 | /* included in all copies or substantial portions of the Software.        */ | 
|---|
| 21 | /*                                                                        */ | 
|---|
| 22 | /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,        */ | 
|---|
| 23 | /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF     */ | 
|---|
| 24 | /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. */ | 
|---|
| 25 | /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY   */ | 
|---|
| 26 | /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,   */ | 
|---|
| 27 | /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE      */ | 
|---|
| 28 | /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.                 */ | 
|---|
| 29 | /**************************************************************************/ | 
|---|
| 30 |  | 
|---|
| 31 | #ifndef WSL_PEER_H | 
|---|
| 32 | #define WSL_PEER_H | 
|---|
| 33 |  | 
|---|
| 34 | #ifndef WEB_ENABLED | 
|---|
| 35 |  | 
|---|
| 36 | #include "packet_buffer.h" | 
|---|
| 37 | #include "websocket_peer.h" | 
|---|
| 38 |  | 
|---|
| 39 | #include "core/crypto/crypto_core.h" | 
|---|
| 40 | #include "core/error/error_list.h" | 
|---|
| 41 | #include "core/io/packet_peer.h" | 
|---|
| 42 | #include "core/io/stream_peer_tcp.h" | 
|---|
| 43 | #include "core/templates/ring_buffer.h" | 
|---|
| 44 |  | 
|---|
| 45 | #include <wslay/wslay.h> | 
|---|
| 46 |  | 
|---|
| 47 | #define  4096 | 
|---|
| 48 |  | 
|---|
| 49 | class WSLPeer : public WebSocketPeer { | 
|---|
| 50 | private: | 
|---|
| 51 | static CryptoCore::RandomGenerator *_static_rng; | 
|---|
| 52 | static WebSocketPeer *_create() { return memnew(WSLPeer); } | 
|---|
| 53 |  | 
|---|
| 54 | // Callbacks. | 
|---|
| 55 | static ssize_t _wsl_recv_callback(wslay_event_context_ptr ctx, uint8_t *data, size_t len, int flags, void *user_data); | 
|---|
| 56 | static ssize_t _wsl_send_callback(wslay_event_context_ptr ctx, const uint8_t *data, size_t len, int flags, void *user_data); | 
|---|
| 57 | static int _wsl_genmask_callback(wslay_event_context_ptr ctx, uint8_t *buf, size_t len, void *user_data); | 
|---|
| 58 | static void _wsl_msg_recv_callback(wslay_event_context_ptr ctx, const struct wslay_event_on_msg_recv_arg *arg, void *user_data); | 
|---|
| 59 |  | 
|---|
| 60 | static wslay_event_callbacks _wsl_callbacks; | 
|---|
| 61 |  | 
|---|
| 62 | // Helpers | 
|---|
| 63 | static String _compute_key_response(String p_key); | 
|---|
| 64 | static String _generate_key(); | 
|---|
| 65 |  | 
|---|
| 66 | // Client IP resolver. | 
|---|
| 67 | class Resolver { | 
|---|
| 68 | Array ip_candidates; | 
|---|
| 69 | IP::ResolverID resolver_id = IP::RESOLVER_INVALID_ID; | 
|---|
| 70 | int port = 0; | 
|---|
| 71 |  | 
|---|
| 72 | public: | 
|---|
| 73 | bool has_more_candidates() { | 
|---|
| 74 | return ip_candidates.size() > 0 || resolver_id != IP::RESOLVER_INVALID_ID; | 
|---|
| 75 | } | 
|---|
| 76 |  | 
|---|
| 77 | void try_next_candidate(Ref<StreamPeerTCP> &p_tcp); | 
|---|
| 78 | void start(const String &p_host, int p_port); | 
|---|
| 79 | void stop(); | 
|---|
| 80 | Resolver() {} | 
|---|
| 81 | }; | 
|---|
| 82 |  | 
|---|
| 83 | Resolver resolver; | 
|---|
| 84 |  | 
|---|
| 85 | // WebSocket connection state. | 
|---|
| 86 | WebSocketPeer::State ready_state = WebSocketPeer::STATE_CLOSED; | 
|---|
| 87 | bool is_server = false; | 
|---|
| 88 | Ref<StreamPeerTCP> tcp; | 
|---|
| 89 | Ref<StreamPeer> connection; | 
|---|
| 90 | wslay_event_context_ptr wsl_ctx = nullptr; | 
|---|
| 91 |  | 
|---|
| 92 | String requested_url; | 
|---|
| 93 | String requested_host; | 
|---|
| 94 | bool pending_request = true; | 
|---|
| 95 | Ref<StreamPeerBuffer> handshake_buffer; | 
|---|
| 96 | String selected_protocol; | 
|---|
| 97 | String session_key; | 
|---|
| 98 |  | 
|---|
| 99 | int close_code = -1; | 
|---|
| 100 | String close_reason; | 
|---|
| 101 | uint8_t was_string = 0; | 
|---|
| 102 |  | 
|---|
| 103 | // WebSocket configuration. | 
|---|
| 104 | bool use_tls = true; | 
|---|
| 105 | Ref<TLSOptions> tls_options; | 
|---|
| 106 |  | 
|---|
| 107 | // Packet buffers. | 
|---|
| 108 | Vector<uint8_t> packet_buffer; | 
|---|
| 109 | // Our packet info is just a boolean (is_string), using uint8_t for it. | 
|---|
| 110 | PacketBuffer<uint8_t> in_buffer; | 
|---|
| 111 |  | 
|---|
| 112 | Error _send(const uint8_t *p_buffer, int p_buffer_size, wslay_opcode p_opcode); | 
|---|
| 113 |  | 
|---|
| 114 | Error _do_server_handshake(); | 
|---|
| 115 | bool _parse_client_request(); | 
|---|
| 116 |  | 
|---|
| 117 | void _do_client_handshake(); | 
|---|
| 118 | bool _verify_server_response(); | 
|---|
| 119 |  | 
|---|
| 120 | void _clear(); | 
|---|
| 121 |  | 
|---|
| 122 | public: | 
|---|
| 123 | static void initialize(); | 
|---|
| 124 | static void deinitialize(); | 
|---|
| 125 |  | 
|---|
| 126 | // PacketPeer | 
|---|
| 127 | virtual int get_available_packet_count() const override; | 
|---|
| 128 | virtual Error get_packet(const uint8_t **r_buffer, int &r_buffer_size) override; | 
|---|
| 129 | virtual Error put_packet(const uint8_t *p_buffer, int p_buffer_size) override; | 
|---|
| 130 | virtual int get_max_packet_size() const override { return packet_buffer.size(); }; | 
|---|
| 131 |  | 
|---|
| 132 | // WebSocketPeer | 
|---|
| 133 | virtual Error send(const uint8_t *p_buffer, int p_buffer_size, WriteMode p_mode) override; | 
|---|
| 134 | virtual Error connect_to_url(const String &p_url, Ref<TLSOptions> p_options = Ref<TLSOptions>()) override; | 
|---|
| 135 | virtual Error accept_stream(Ref<StreamPeer> p_stream) override; | 
|---|
| 136 | virtual void close(int p_code = 1000, String p_reason = "") override; | 
|---|
| 137 | virtual void poll() override; | 
|---|
| 138 |  | 
|---|
| 139 | virtual State get_ready_state() const override { return ready_state; } | 
|---|
| 140 | virtual int get_close_code() const override { return close_code; } | 
|---|
| 141 | virtual String get_close_reason() const override { return close_reason; } | 
|---|
| 142 | virtual int get_current_outbound_buffered_amount() const override; | 
|---|
| 143 |  | 
|---|
| 144 | virtual IPAddress get_connected_host() const override; | 
|---|
| 145 | virtual uint16_t get_connected_port() const override; | 
|---|
| 146 | virtual String get_selected_protocol() const override; | 
|---|
| 147 | virtual String get_requested_url() const override; | 
|---|
| 148 |  | 
|---|
| 149 | virtual bool was_string_packet() const override { return was_string; } | 
|---|
| 150 | virtual void set_no_delay(bool p_enabled) override; | 
|---|
| 151 |  | 
|---|
| 152 | WSLPeer(); | 
|---|
| 153 | ~WSLPeer(); | 
|---|
| 154 | }; | 
|---|
| 155 |  | 
|---|
| 156 | #endif // WEB_ENABLED | 
|---|
| 157 |  | 
|---|
| 158 | #endif // WSL_PEER_H | 
|---|
| 159 |  | 
|---|