2016-06-18 14:46:12 +02:00
|
|
|
/*************************************************************************/
|
|
|
|
/* stream_peer_ssl.cpp */
|
|
|
|
/*************************************************************************/
|
|
|
|
/* This file is part of: */
|
|
|
|
/* GODOT ENGINE */
|
2017-08-27 14:16:55 +02:00
|
|
|
/* https://godotengine.org */
|
2016-06-18 14:46:12 +02:00
|
|
|
/*************************************************************************/
|
2019-01-01 12:53:14 +01:00
|
|
|
/* Copyright (c) 2007-2019 Juan Linietsky, Ariel Manzur. */
|
|
|
|
/* Copyright (c) 2014-2019 Godot Engine contributors (cf. AUTHORS.md) */
|
2016-06-18 14:46:12 +02:00
|
|
|
/* */
|
|
|
|
/* 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. */
|
|
|
|
/*************************************************************************/
|
2018-01-05 00:50:27 +01:00
|
|
|
|
2014-04-29 02:56:43 +02:00
|
|
|
#include "stream_peer_ssl.h"
|
2018-09-11 18:13:45 +02:00
|
|
|
|
2018-09-14 16:02:04 +02:00
|
|
|
#include "core/io/certs_compressed.gen.h"
|
|
|
|
#include "core/io/compression.h"
|
2018-09-11 18:13:45 +02:00
|
|
|
#include "core/os/file_access.h"
|
|
|
|
#include "core/project_settings.h"
|
2014-04-29 02:56:43 +02:00
|
|
|
|
2017-03-05 16:44:50 +01:00
|
|
|
StreamPeerSSL *(*StreamPeerSSL::_create)() = NULL;
|
2016-03-12 14:44:12 +01:00
|
|
|
|
2014-04-29 02:56:43 +02:00
|
|
|
StreamPeerSSL *StreamPeerSSL::create() {
|
|
|
|
|
2019-07-02 15:28:34 +02:00
|
|
|
if (_create)
|
|
|
|
return _create();
|
|
|
|
return NULL;
|
2014-04-29 02:56:43 +02:00
|
|
|
}
|
|
|
|
|
2017-03-05 16:44:50 +01:00
|
|
|
StreamPeerSSL::LoadCertsFromMemory StreamPeerSSL::load_certs_func = NULL;
|
|
|
|
bool StreamPeerSSL::available = false;
|
2014-04-29 02:56:43 +02:00
|
|
|
|
2017-03-05 16:44:50 +01:00
|
|
|
void StreamPeerSSL::load_certs_from_memory(const PoolByteArray &p_memory) {
|
2016-03-12 14:44:12 +01:00
|
|
|
if (load_certs_func)
|
|
|
|
load_certs_func(p_memory);
|
|
|
|
}
|
|
|
|
|
2018-09-15 14:45:54 +02:00
|
|
|
void StreamPeerSSL::load_certs_from_file(String p_path) {
|
|
|
|
if (p_path != "") {
|
|
|
|
PoolByteArray certs = get_cert_file_as_array(p_path);
|
|
|
|
if (certs.size() > 0)
|
|
|
|
load_certs_func(certs);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-03-12 14:44:12 +01:00
|
|
|
bool StreamPeerSSL::is_available() {
|
|
|
|
return available;
|
|
|
|
}
|
|
|
|
|
2018-07-14 22:33:30 +02:00
|
|
|
void StreamPeerSSL::set_blocking_handshake_enabled(bool p_enabled) {
|
|
|
|
blocking_handshake = p_enabled;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool StreamPeerSSL::is_blocking_handshake_enabled() const {
|
|
|
|
return blocking_handshake;
|
|
|
|
}
|
|
|
|
|
2018-09-15 14:45:54 +02:00
|
|
|
PoolByteArray StreamPeerSSL::get_cert_file_as_array(String p_path) {
|
|
|
|
|
|
|
|
PoolByteArray out;
|
|
|
|
FileAccess *f = FileAccess::open(p_path, FileAccess::READ);
|
|
|
|
if (f) {
|
|
|
|
int flen = f->get_len();
|
|
|
|
out.resize(flen + 1);
|
|
|
|
PoolByteArray::Write w = out.write();
|
|
|
|
f->get_buffer(w.ptr(), flen);
|
|
|
|
w[flen] = 0; // Make sure it ends with string terminator
|
|
|
|
memdelete(f);
|
|
|
|
#ifdef DEBUG_ENABLED
|
|
|
|
print_verbose(vformat("Loaded certs from '%s'.", p_path));
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
2018-03-28 16:26:25 +02:00
|
|
|
PoolByteArray StreamPeerSSL::get_project_cert_array() {
|
|
|
|
|
|
|
|
PoolByteArray out;
|
|
|
|
String certs_path = GLOBAL_DEF("network/ssl/certificates", "");
|
|
|
|
ProjectSettings::get_singleton()->set_custom_property_info("network/ssl/certificates", PropertyInfo(Variant::STRING, "network/ssl/certificates", PROPERTY_HINT_FILE, "*.crt"));
|
|
|
|
|
|
|
|
if (certs_path != "") {
|
2018-09-14 16:02:04 +02:00
|
|
|
// Use certs defined in project settings.
|
2018-09-15 14:45:54 +02:00
|
|
|
return get_cert_file_as_array(certs_path);
|
2018-03-28 16:26:25 +02:00
|
|
|
}
|
2018-09-14 16:02:04 +02:00
|
|
|
#ifdef BUILTIN_CERTS_ENABLED
|
|
|
|
else {
|
|
|
|
// Use builtin certs only if user did not override it in project settings.
|
|
|
|
out.resize(_certs_uncompressed_size + 1);
|
|
|
|
PoolByteArray::Write w = out.write();
|
|
|
|
Compression::decompress(w.ptr(), _certs_uncompressed_size, _certs_compressed, _certs_compressed_size, Compression::MODE_DEFLATE);
|
|
|
|
w[_certs_uncompressed_size] = 0; // Make sure it ends with string terminator
|
|
|
|
#ifdef DEBUG_ENABLED
|
|
|
|
print_verbose("Loaded builtin certs");
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#endif
|
2018-03-28 16:26:25 +02:00
|
|
|
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
2014-04-29 02:56:43 +02:00
|
|
|
void StreamPeerSSL::_bind_methods() {
|
|
|
|
|
2018-02-09 15:03:34 +01:00
|
|
|
ClassDB::bind_method(D_METHOD("poll"), &StreamPeerSSL::poll);
|
2018-07-26 11:56:21 +02:00
|
|
|
ClassDB::bind_method(D_METHOD("accept_stream", "base"), &StreamPeerSSL::accept_stream);
|
2017-08-09 13:19:41 +02:00
|
|
|
ClassDB::bind_method(D_METHOD("connect_to_stream", "stream", "validate_certs", "for_hostname"), &StreamPeerSSL::connect_to_stream, DEFVAL(false), DEFVAL(String()));
|
2017-03-05 16:44:50 +01:00
|
|
|
ClassDB::bind_method(D_METHOD("get_status"), &StreamPeerSSL::get_status);
|
|
|
|
ClassDB::bind_method(D_METHOD("disconnect_from_stream"), &StreamPeerSSL::disconnect_from_stream);
|
2018-07-14 22:33:30 +02:00
|
|
|
ClassDB::bind_method(D_METHOD("set_blocking_handshake_enabled", "enabled"), &StreamPeerSSL::set_blocking_handshake_enabled);
|
|
|
|
ClassDB::bind_method(D_METHOD("is_blocking_handshake_enabled"), &StreamPeerSSL::is_blocking_handshake_enabled);
|
|
|
|
|
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::BOOL, "blocking_handshake"), "set_blocking_handshake_enabled", "is_blocking_handshake_enabled");
|
2017-08-20 17:45:01 +02:00
|
|
|
|
|
|
|
BIND_ENUM_CONSTANT(STATUS_DISCONNECTED);
|
2018-09-14 14:13:11 +02:00
|
|
|
BIND_ENUM_CONSTANT(STATUS_HANDSHAKING);
|
2017-08-20 17:45:01 +02:00
|
|
|
BIND_ENUM_CONSTANT(STATUS_CONNECTED);
|
2018-07-14 22:33:30 +02:00
|
|
|
BIND_ENUM_CONSTANT(STATUS_ERROR);
|
2017-08-20 17:45:01 +02:00
|
|
|
BIND_ENUM_CONSTANT(STATUS_ERROR_HOSTNAME_MISMATCH);
|
2014-04-29 02:56:43 +02:00
|
|
|
}
|
|
|
|
|
2017-03-05 16:44:50 +01:00
|
|
|
StreamPeerSSL::StreamPeerSSL() {
|
2018-07-14 22:33:30 +02:00
|
|
|
blocking_handshake = true;
|
2014-04-29 02:56:43 +02:00
|
|
|
}
|