|
@@ -41,13 +41,13 @@ VARIANT_ENUM_CAST(IP::ResolverStatus);
|
|
|
struct _IP_ResolverPrivate {
|
|
|
struct QueueItem {
|
|
|
SafeNumeric<IP::ResolverStatus> status;
|
|
|
- IP_Address response;
|
|
|
+ List<IP_Address> response;
|
|
|
String hostname;
|
|
|
IP::Type type;
|
|
|
|
|
|
void clear() {
|
|
|
status.set(IP::RESOLVER_STATUS_NONE);
|
|
|
- response = IP_Address();
|
|
|
+ response.clear();
|
|
|
type = IP::TYPE_NONE;
|
|
|
hostname = "";
|
|
|
};
|
|
@@ -80,13 +80,9 @@ struct _IP_ResolverPrivate {
|
|
|
if (queue[i].status.get() != IP::RESOLVER_STATUS_WAITING) {
|
|
|
continue;
|
|
|
}
|
|
|
- queue[i].response = IP::get_singleton()->resolve_hostname(queue[i].hostname, queue[i].type);
|
|
|
|
|
|
- if (!queue[i].response.is_valid()) {
|
|
|
- queue[i].status.set(IP::RESOLVER_STATUS_ERROR);
|
|
|
- } else {
|
|
|
- queue[i].status.set(IP::RESOLVER_STATUS_DONE);
|
|
|
- }
|
|
|
+ IP::get_singleton()->_resolve_hostname(queue[i].response, queue[i].hostname, queue[i].type);
|
|
|
+ queue[i].status.set(queue[i].response.empty() ? IP::RESOLVER_STATUS_ERROR : IP::RESOLVER_STATUS_DONE);
|
|
|
}
|
|
|
}
|
|
|
|
|
@@ -96,13 +92,12 @@ struct _IP_ResolverPrivate {
|
|
|
while (!ipr->thread_abort) {
|
|
|
ipr->sem.wait();
|
|
|
|
|
|
- ipr->mutex.lock();
|
|
|
+ MutexLock lock(ipr->mutex);
|
|
|
ipr->resolve_queues();
|
|
|
- ipr->mutex.unlock();
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- HashMap<String, IP_Address> cache;
|
|
|
+ HashMap<String, List<IP_Address>> cache;
|
|
|
|
|
|
static String get_cache_key(String p_hostname, IP::Type p_type) {
|
|
|
return itos(p_type) + p_hostname;
|
|
@@ -110,40 +105,63 @@ struct _IP_ResolverPrivate {
|
|
|
};
|
|
|
|
|
|
IP_Address IP::resolve_hostname(const String &p_hostname, IP::Type p_type) {
|
|
|
- resolver->mutex.lock();
|
|
|
+ MutexLock lock(resolver->mutex);
|
|
|
+
|
|
|
+ List<IP_Address> res;
|
|
|
|
|
|
String key = _IP_ResolverPrivate::get_cache_key(p_hostname, p_type);
|
|
|
- if (resolver->cache.has(key) && resolver->cache[key].is_valid()) {
|
|
|
- IP_Address res = resolver->cache[key];
|
|
|
- resolver->mutex.unlock();
|
|
|
- return res;
|
|
|
+ if (resolver->cache.has(key)) {
|
|
|
+ res = resolver->cache[key];
|
|
|
+ } else {
|
|
|
+ _resolve_hostname(res, p_hostname, p_type);
|
|
|
+ resolver->cache[key] = res;
|
|
|
}
|
|
|
|
|
|
- IP_Address res = _resolve_hostname(p_hostname, p_type);
|
|
|
- resolver->cache[key] = res;
|
|
|
- resolver->mutex.unlock();
|
|
|
- return res;
|
|
|
+ for (int i = 0; i < res.size(); ++i) {
|
|
|
+ if (res[i].is_valid()) {
|
|
|
+ return res[i];
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return IP_Address();
|
|
|
+}
|
|
|
+
|
|
|
+Array IP::resolve_hostname_addresses(const String &p_hostname, Type p_type) {
|
|
|
+ MutexLock lock(resolver->mutex);
|
|
|
+
|
|
|
+ String key = _IP_ResolverPrivate::get_cache_key(p_hostname, p_type);
|
|
|
+ if (!resolver->cache.has(key)) {
|
|
|
+ _resolve_hostname(resolver->cache[key], p_hostname, p_type);
|
|
|
+ }
|
|
|
+
|
|
|
+ List<IP_Address> res = resolver->cache[key];
|
|
|
+
|
|
|
+ Array result;
|
|
|
+ for (int i = 0; i < res.size(); ++i) {
|
|
|
+ if (res[i].is_valid()) {
|
|
|
+ result.push_back(String(res[i]));
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return result;
|
|
|
}
|
|
|
|
|
|
IP::ResolverID IP::resolve_hostname_queue_item(const String &p_hostname, IP::Type p_type) {
|
|
|
- resolver->mutex.lock();
|
|
|
+ MutexLock lock(resolver->mutex);
|
|
|
|
|
|
ResolverID id = resolver->find_empty_id();
|
|
|
|
|
|
if (id == RESOLVER_INVALID_ID) {
|
|
|
WARN_PRINT("Out of resolver queries");
|
|
|
- resolver->mutex.unlock();
|
|
|
return id;
|
|
|
}
|
|
|
|
|
|
String key = _IP_ResolverPrivate::get_cache_key(p_hostname, p_type);
|
|
|
resolver->queue[id].hostname = p_hostname;
|
|
|
resolver->queue[id].type = p_type;
|
|
|
- if (resolver->cache.has(key) && resolver->cache[key].is_valid()) {
|
|
|
+ if (resolver->cache.has(key)) {
|
|
|
resolver->queue[id].response = resolver->cache[key];
|
|
|
resolver->queue[id].status.set(IP::RESOLVER_STATUS_DONE);
|
|
|
} else {
|
|
|
- resolver->queue[id].response = IP_Address();
|
|
|
+ resolver->queue[id].response = List<IP_Address>();
|
|
|
resolver->queue[id].status.set(IP::RESOLVER_STATUS_WAITING);
|
|
|
if (resolver->thread.is_started()) {
|
|
|
resolver->sem.post();
|
|
@@ -152,54 +170,74 @@ IP::ResolverID IP::resolve_hostname_queue_item(const String &p_hostname, IP::Typ
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- resolver->mutex.unlock();
|
|
|
return id;
|
|
|
}
|
|
|
|
|
|
IP::ResolverStatus IP::get_resolve_item_status(ResolverID p_id) const {
|
|
|
ERR_FAIL_INDEX_V(p_id, IP::RESOLVER_MAX_QUERIES, IP::RESOLVER_STATUS_NONE);
|
|
|
|
|
|
- resolver->mutex.lock();
|
|
|
+ MutexLock lock(resolver->mutex);
|
|
|
+
|
|
|
if (resolver->queue[p_id].status.get() == IP::RESOLVER_STATUS_NONE) {
|
|
|
ERR_PRINT("Condition status == IP::RESOLVER_STATUS_NONE");
|
|
|
- resolver->mutex.unlock();
|
|
|
return IP::RESOLVER_STATUS_NONE;
|
|
|
}
|
|
|
IP::ResolverStatus res = resolver->queue[p_id].status.get();
|
|
|
|
|
|
- resolver->mutex.unlock();
|
|
|
return res;
|
|
|
}
|
|
|
|
|
|
IP_Address IP::get_resolve_item_address(ResolverID p_id) const {
|
|
|
ERR_FAIL_INDEX_V(p_id, IP::RESOLVER_MAX_QUERIES, IP_Address());
|
|
|
|
|
|
- resolver->mutex.lock();
|
|
|
+ MutexLock lock(resolver->mutex);
|
|
|
|
|
|
if (resolver->queue[p_id].status.get() != IP::RESOLVER_STATUS_DONE) {
|
|
|
ERR_PRINTS("Resolve of '" + resolver->queue[p_id].hostname + "'' didn't complete yet.");
|
|
|
- resolver->mutex.unlock();
|
|
|
return IP_Address();
|
|
|
}
|
|
|
|
|
|
- IP_Address res = resolver->queue[p_id].response;
|
|
|
+ List<IP_Address> res = resolver->queue[p_id].response;
|
|
|
|
|
|
- resolver->mutex.unlock();
|
|
|
- return res;
|
|
|
+ for (int i = 0; i < res.size(); ++i) {
|
|
|
+ if (res[i].is_valid()) {
|
|
|
+ return res[i];
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return IP_Address();
|
|
|
+}
|
|
|
+
|
|
|
+Array IP::get_resolve_item_addresses(ResolverID p_id) const {
|
|
|
+ ERR_FAIL_INDEX_V(p_id, IP::RESOLVER_MAX_QUERIES, Array());
|
|
|
+
|
|
|
+ MutexLock lock(resolver->mutex);
|
|
|
+
|
|
|
+ if (resolver->queue[p_id].status.get() != IP::RESOLVER_STATUS_DONE) {
|
|
|
+ ERR_PRINTS("Resolve of '" + resolver->queue[p_id].hostname + "'' didn't complete yet.");
|
|
|
+ return Array();
|
|
|
+ }
|
|
|
+
|
|
|
+ List<IP_Address> res = resolver->queue[p_id].response;
|
|
|
+
|
|
|
+ Array result;
|
|
|
+ for (int i = 0; i < res.size(); ++i) {
|
|
|
+ if (res[i].is_valid()) {
|
|
|
+ result.push_back(String(res[i]));
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return result;
|
|
|
}
|
|
|
|
|
|
void IP::erase_resolve_item(ResolverID p_id) {
|
|
|
ERR_FAIL_INDEX(p_id, IP::RESOLVER_MAX_QUERIES);
|
|
|
|
|
|
- resolver->mutex.lock();
|
|
|
+ MutexLock lock(resolver->mutex);
|
|
|
|
|
|
resolver->queue[p_id].status.set(IP::RESOLVER_STATUS_NONE);
|
|
|
-
|
|
|
- resolver->mutex.unlock();
|
|
|
}
|
|
|
|
|
|
void IP::clear_cache(const String &p_hostname) {
|
|
|
- resolver->mutex.lock();
|
|
|
+ MutexLock lock(resolver->mutex);
|
|
|
|
|
|
if (p_hostname.empty()) {
|
|
|
resolver->cache.clear();
|
|
@@ -209,8 +247,6 @@ void IP::clear_cache(const String &p_hostname) {
|
|
|
resolver->cache.erase(_IP_ResolverPrivate::get_cache_key(p_hostname, IP::TYPE_IPV6));
|
|
|
resolver->cache.erase(_IP_ResolverPrivate::get_cache_key(p_hostname, IP::TYPE_ANY));
|
|
|
}
|
|
|
-
|
|
|
- resolver->mutex.unlock();
|
|
|
}
|
|
|
|
|
|
Array IP::_get_local_addresses() const {
|
|
@@ -259,9 +295,11 @@ void IP::get_local_addresses(List<IP_Address> *r_addresses) const {
|
|
|
|
|
|
void IP::_bind_methods() {
|
|
|
ClassDB::bind_method(D_METHOD("resolve_hostname", "host", "ip_type"), &IP::resolve_hostname, DEFVAL(IP::TYPE_ANY));
|
|
|
+ ClassDB::bind_method(D_METHOD("resolve_hostname_addresses", "host", "ip_type"), &IP::resolve_hostname_addresses, DEFVAL(IP::TYPE_ANY));
|
|
|
ClassDB::bind_method(D_METHOD("resolve_hostname_queue_item", "host", "ip_type"), &IP::resolve_hostname_queue_item, DEFVAL(IP::TYPE_ANY));
|
|
|
ClassDB::bind_method(D_METHOD("get_resolve_item_status", "id"), &IP::get_resolve_item_status);
|
|
|
ClassDB::bind_method(D_METHOD("get_resolve_item_address", "id"), &IP::get_resolve_item_address);
|
|
|
+ ClassDB::bind_method(D_METHOD("get_resolve_item_addresses", "id"), &IP::get_resolve_item_addresses);
|
|
|
ClassDB::bind_method(D_METHOD("erase_resolve_item", "id"), &IP::erase_resolve_item);
|
|
|
ClassDB::bind_method(D_METHOD("get_local_addresses"), &IP::_get_local_addresses);
|
|
|
ClassDB::bind_method(D_METHOD("get_local_interfaces"), &IP::_get_local_interfaces);
|