| 
									
										
										
										
											2023-01-05 13:25:55 +01:00
										 |  |  | /**************************************************************************/ | 
					
						
							|  |  |  | /*  editor_http_server.h                                                  */ | 
					
						
							|  |  |  | /**************************************************************************/ | 
					
						
							|  |  |  | /*                         This file is part of:                          */ | 
					
						
							|  |  |  | /*                             GODOT ENGINE                               */ | 
					
						
							|  |  |  | /*                        https://godotengine.org                         */ | 
					
						
							|  |  |  | /**************************************************************************/ | 
					
						
							|  |  |  | /* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */ | 
					
						
							|  |  |  | /* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur.                  */ | 
					
						
							|  |  |  | /*                                                                        */ | 
					
						
							|  |  |  | /* Permission is hereby granted, free of charge, to any person obtaining  */ | 
					
						
							|  |  |  | /* a copy of this software and associated documentation files (the        */ | 
					
						
							|  |  |  | /* "Software"), to deal in the Software without restriction, including    */ | 
					
						
							|  |  |  | /* without limitation the rights to use, copy, modify, merge, publish,    */ | 
					
						
							|  |  |  | /* distribute, sublicense, and/or sell copies of the Software, and to     */ | 
					
						
							|  |  |  | /* permit persons to whom the Software is furnished to do so, subject to  */ | 
					
						
							|  |  |  | /* the following conditions:                                              */ | 
					
						
							|  |  |  | /*                                                                        */ | 
					
						
							|  |  |  | /* The above copyright notice and this permission notice shall be         */ | 
					
						
							|  |  |  | /* included in all copies or substantial portions of the Software.        */ | 
					
						
							|  |  |  | /*                                                                        */ | 
					
						
							|  |  |  | /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,        */ | 
					
						
							|  |  |  | /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF     */ | 
					
						
							|  |  |  | /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. */ | 
					
						
							|  |  |  | /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY   */ | 
					
						
							|  |  |  | /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,   */ | 
					
						
							|  |  |  | /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE      */ | 
					
						
							|  |  |  | /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.                 */ | 
					
						
							|  |  |  | /**************************************************************************/ | 
					
						
							| 
									
										
										
										
											2021-07-14 23:16:48 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-28 20:27:45 +02:00
										 |  |  | #ifndef WEB_EDITOR_HTTP_SERVER_H
 | 
					
						
							|  |  |  | #define WEB_EDITOR_HTTP_SERVER_H
 | 
					
						
							| 
									
										
										
										
											2021-07-14 23:16:48 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "core/io/image_loader.h"
 | 
					
						
							| 
									
										
										
										
											2022-09-07 00:46:12 +02:00
										 |  |  | #include "core/io/stream_peer_tls.h"
 | 
					
						
							| 
									
										
										
										
											2021-07-14 23:16:48 +03:00
										 |  |  | #include "core/io/tcp_server.h"
 | 
					
						
							|  |  |  | #include "core/io/zip_io.h"
 | 
					
						
							| 
									
										
										
										
											2022-02-14 14:00:03 +01:00
										 |  |  | #include "editor/editor_paths.h"
 | 
					
						
							| 
									
										
										
										
											2021-07-14 23:16:48 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | class EditorHTTPServer : public RefCounted { | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  | 	Ref<TCPServer> server; | 
					
						
							| 
									
										
										
										
											2022-05-13 15:04:37 +02:00
										 |  |  | 	HashMap<String, String> mimes; | 
					
						
							| 
									
										
										
										
											2021-07-14 23:16:48 +03:00
										 |  |  | 	Ref<StreamPeerTCP> tcp; | 
					
						
							| 
									
										
										
										
											2022-09-07 08:25:47 +02:00
										 |  |  | 	Ref<StreamPeerTLS> tls; | 
					
						
							| 
									
										
										
										
											2021-07-14 23:16:48 +03:00
										 |  |  | 	Ref<StreamPeer> peer; | 
					
						
							|  |  |  | 	Ref<CryptoKey> key; | 
					
						
							|  |  |  | 	Ref<X509Certificate> cert; | 
					
						
							| 
									
										
										
										
											2022-09-07 08:25:47 +02:00
										 |  |  | 	bool use_tls = false; | 
					
						
							| 
									
										
										
										
											2021-07-14 23:16:48 +03:00
										 |  |  | 	uint64_t time = 0; | 
					
						
							|  |  |  | 	uint8_t req_buf[4096]; | 
					
						
							|  |  |  | 	int req_pos = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	void _clear_client() { | 
					
						
							|  |  |  | 		peer = Ref<StreamPeer>(); | 
					
						
							| 
									
										
										
										
											2022-09-07 08:25:47 +02:00
										 |  |  | 		tls = Ref<StreamPeerTLS>(); | 
					
						
							| 
									
										
										
										
											2021-07-14 23:16:48 +03:00
										 |  |  | 		tcp = Ref<StreamPeerTCP>(); | 
					
						
							|  |  |  | 		memset(req_buf, 0, sizeof(req_buf)); | 
					
						
							|  |  |  | 		time = 0; | 
					
						
							|  |  |  | 		req_pos = 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	void _set_internal_certs(Ref<Crypto> p_crypto) { | 
					
						
							|  |  |  | 		const String cache_path = EditorPaths::get_singleton()->get_cache_dir(); | 
					
						
							| 
									
										
										
										
											2022-08-29 19:34:01 -05:00
										 |  |  | 		const String key_path = cache_path.path_join("html5_server.key"); | 
					
						
							|  |  |  | 		const String crt_path = cache_path.path_join("html5_server.crt"); | 
					
						
							| 
									
										
										
										
											2021-07-14 23:16:48 +03:00
										 |  |  | 		bool regen = !FileAccess::exists(key_path) || !FileAccess::exists(crt_path); | 
					
						
							|  |  |  | 		if (!regen) { | 
					
						
							|  |  |  | 			key = Ref<CryptoKey>(CryptoKey::create()); | 
					
						
							|  |  |  | 			cert = Ref<X509Certificate>(X509Certificate::create()); | 
					
						
							|  |  |  | 			if (key->load(key_path) != OK || cert->load(crt_path) != OK) { | 
					
						
							|  |  |  | 				regen = true; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (regen) { | 
					
						
							|  |  |  | 			key = p_crypto->generate_rsa(2048); | 
					
						
							|  |  |  | 			key->save(key_path); | 
					
						
							|  |  |  | 			cert = p_crypto->generate_self_signed_certificate(key, "CN=godot-debug.local,O=A Game Dev,C=XXA", "20140101000000", "20340101000000"); | 
					
						
							|  |  |  | 			cert->save(crt_path); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  | 	EditorHTTPServer() { | 
					
						
							|  |  |  | 		mimes["html"] = "text/html"; | 
					
						
							|  |  |  | 		mimes["js"] = "application/javascript"; | 
					
						
							|  |  |  | 		mimes["json"] = "application/json"; | 
					
						
							|  |  |  | 		mimes["pck"] = "application/octet-stream"; | 
					
						
							|  |  |  | 		mimes["png"] = "image/png"; | 
					
						
							|  |  |  | 		mimes["svg"] = "image/svg"; | 
					
						
							|  |  |  | 		mimes["wasm"] = "application/wasm"; | 
					
						
							|  |  |  | 		server.instantiate(); | 
					
						
							|  |  |  | 		stop(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	void stop() { | 
					
						
							|  |  |  | 		server->stop(); | 
					
						
							|  |  |  | 		_clear_client(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-07 08:25:47 +02:00
										 |  |  | 	Error listen(int p_port, IPAddress p_address, bool p_use_tls, String p_tls_key, String p_tls_cert) { | 
					
						
							|  |  |  | 		use_tls = p_use_tls; | 
					
						
							|  |  |  | 		if (use_tls) { | 
					
						
							| 
									
										
										
										
											2021-07-14 23:16:48 +03:00
										 |  |  | 			Ref<Crypto> crypto = Crypto::create(); | 
					
						
							|  |  |  | 			if (crypto.is_null()) { | 
					
						
							|  |  |  | 				return ERR_UNAVAILABLE; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2022-09-07 08:25:47 +02:00
										 |  |  | 			if (!p_tls_key.is_empty() && !p_tls_cert.is_empty()) { | 
					
						
							| 
									
										
										
										
											2021-07-14 23:16:48 +03:00
										 |  |  | 				key = Ref<CryptoKey>(CryptoKey::create()); | 
					
						
							| 
									
										
										
										
											2022-09-07 08:25:47 +02:00
										 |  |  | 				Error err = key->load(p_tls_key); | 
					
						
							| 
									
										
										
										
											2021-07-14 23:16:48 +03:00
										 |  |  | 				ERR_FAIL_COND_V(err != OK, err); | 
					
						
							|  |  |  | 				cert = Ref<X509Certificate>(X509Certificate::create()); | 
					
						
							| 
									
										
										
										
											2022-09-07 08:25:47 +02:00
										 |  |  | 				err = cert->load(p_tls_cert); | 
					
						
							| 
									
										
										
										
											2021-07-14 23:16:48 +03:00
										 |  |  | 				ERR_FAIL_COND_V(err != OK, err); | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				_set_internal_certs(crypto); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return server->listen(p_port, p_address); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	bool is_listening() const { | 
					
						
							|  |  |  | 		return server->is_listening(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	void _send_response() { | 
					
						
							|  |  |  | 		Vector<String> psa = String((char *)req_buf).split("\r\n"); | 
					
						
							|  |  |  | 		int len = psa.size(); | 
					
						
							|  |  |  | 		ERR_FAIL_COND_MSG(len < 4, "Not enough response headers, got: " + itos(len) + ", expected >= 4."); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Vector<String> req = psa[0].split(" ", false); | 
					
						
							|  |  |  | 		ERR_FAIL_COND_MSG(req.size() < 2, "Invalid protocol or status code."); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Wrong protocol
 | 
					
						
							|  |  |  | 		ERR_FAIL_COND_MSG(req[0] != "GET" || req[2] != "HTTP/1.1", "Invalid method or HTTP version."); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		const int query_index = req[1].find_char('?'); | 
					
						
							|  |  |  | 		const String path = (query_index == -1) ? req[1] : req[1].substr(0, query_index); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		const String req_file = path.get_file(); | 
					
						
							|  |  |  | 		const String req_ext = path.get_extension(); | 
					
						
							| 
									
										
										
										
											2022-08-29 19:34:01 -05:00
										 |  |  | 		const String cache_path = EditorPaths::get_singleton()->get_cache_dir().path_join("web"); | 
					
						
							|  |  |  | 		const String filepath = cache_path.path_join(req_file); | 
					
						
							| 
									
										
										
										
											2021-07-14 23:16:48 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		if (!mimes.has(req_ext) || !FileAccess::exists(filepath)) { | 
					
						
							|  |  |  | 			String s = "HTTP/1.1 404 Not Found\r\n"; | 
					
						
							|  |  |  | 			s += "Connection: Close\r\n"; | 
					
						
							|  |  |  | 			s += "\r\n"; | 
					
						
							|  |  |  | 			CharString cs = s.utf8(); | 
					
						
							|  |  |  | 			peer->put_data((const uint8_t *)cs.get_data(), cs.size() - 1); | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		const String ctype = mimes[req_ext]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-23 11:08:58 +02:00
										 |  |  | 		Ref<FileAccess> f = FileAccess::open(filepath, FileAccess::READ); | 
					
						
							|  |  |  | 		ERR_FAIL_COND(f.is_null()); | 
					
						
							| 
									
										
										
										
											2021-07-14 23:16:48 +03:00
										 |  |  | 		String s = "HTTP/1.1 200 OK\r\n"; | 
					
						
							|  |  |  | 		s += "Connection: Close\r\n"; | 
					
						
							|  |  |  | 		s += "Content-Type: " + ctype + "\r\n"; | 
					
						
							|  |  |  | 		s += "Access-Control-Allow-Origin: *\r\n"; | 
					
						
							|  |  |  | 		s += "Cross-Origin-Opener-Policy: same-origin\r\n"; | 
					
						
							|  |  |  | 		s += "Cross-Origin-Embedder-Policy: require-corp\r\n"; | 
					
						
							|  |  |  | 		s += "Cache-Control: no-store, max-age=0\r\n"; | 
					
						
							|  |  |  | 		s += "\r\n"; | 
					
						
							|  |  |  | 		CharString cs = s.utf8(); | 
					
						
							|  |  |  | 		Error err = peer->put_data((const uint8_t *)cs.get_data(), cs.size() - 1); | 
					
						
							|  |  |  | 		if (err != OK) { | 
					
						
							|  |  |  | 			ERR_FAIL(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		while (true) { | 
					
						
							|  |  |  | 			uint8_t bytes[4096]; | 
					
						
							|  |  |  | 			uint64_t read = f->get_buffer(bytes, 4096); | 
					
						
							|  |  |  | 			if (read == 0) { | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			err = peer->put_data(bytes, read); | 
					
						
							|  |  |  | 			if (err != OK) { | 
					
						
							|  |  |  | 				ERR_FAIL(); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	void poll() { | 
					
						
							|  |  |  | 		if (!server->is_listening()) { | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (tcp.is_null()) { | 
					
						
							|  |  |  | 			if (!server->is_connection_available()) { | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			tcp = server->take_connection(); | 
					
						
							|  |  |  | 			peer = tcp; | 
					
						
							|  |  |  | 			time = OS::get_singleton()->get_ticks_usec(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (OS::get_singleton()->get_ticks_usec() - time > 1000000) { | 
					
						
							|  |  |  | 			_clear_client(); | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (tcp->get_status() != StreamPeerTCP::STATUS_CONNECTED) { | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-07 08:25:47 +02:00
										 |  |  | 		if (use_tls) { | 
					
						
							|  |  |  | 			if (tls.is_null()) { | 
					
						
							|  |  |  | 				tls = Ref<StreamPeerTLS>(StreamPeerTLS::create()); | 
					
						
							|  |  |  | 				peer = tls; | 
					
						
							|  |  |  | 				tls->set_blocking_handshake_enabled(false); | 
					
						
							|  |  |  | 				if (tls->accept_stream(tcp, key, cert) != OK) { | 
					
						
							| 
									
										
										
										
											2021-07-14 23:16:48 +03:00
										 |  |  | 					_clear_client(); | 
					
						
							|  |  |  | 					return; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2022-09-07 08:25:47 +02:00
										 |  |  | 			tls->poll(); | 
					
						
							|  |  |  | 			if (tls->get_status() == StreamPeerTLS::STATUS_HANDSHAKING) { | 
					
						
							| 
									
										
										
										
											2021-07-14 23:16:48 +03:00
										 |  |  | 				// Still handshaking, keep waiting.
 | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2022-09-07 08:25:47 +02:00
										 |  |  | 			if (tls->get_status() != StreamPeerTLS::STATUS_CONNECTED) { | 
					
						
							| 
									
										
										
										
											2021-07-14 23:16:48 +03:00
										 |  |  | 				_clear_client(); | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		while (true) { | 
					
						
							|  |  |  | 			char *r = (char *)req_buf; | 
					
						
							|  |  |  | 			int l = req_pos - 1; | 
					
						
							|  |  |  | 			if (l > 3 && r[l] == '\n' && r[l - 1] == '\r' && r[l - 2] == '\n' && r[l - 3] == '\r') { | 
					
						
							|  |  |  | 				_send_response(); | 
					
						
							|  |  |  | 				_clear_client(); | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			int read = 0; | 
					
						
							|  |  |  | 			ERR_FAIL_COND(req_pos >= 4096); | 
					
						
							|  |  |  | 			Error err = peer->get_partial_data(&req_buf[req_pos], 1, read); | 
					
						
							|  |  |  | 			if (err != OK) { | 
					
						
							|  |  |  | 				// Got an error
 | 
					
						
							|  |  |  | 				_clear_client(); | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 			} else if (read != 1) { | 
					
						
							|  |  |  | 				// Busy, wait next poll
 | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			req_pos += read; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-28 20:27:45 +02:00
										 |  |  | #endif // WEB_EDITOR_HTTP_SERVER_H
 |