Features/call reporting timeout (#1)

* Timed out and canceled netreqs are finished too

* Minor code duplication elemination

* Blah typo

* Embarrassing logic error
This commit is contained in:
wtoorop 2016-06-23 14:02:55 +02:00 committed by Robert Groenenberg
parent 60c6c8d8ca
commit a435932b04
5 changed files with 28 additions and 56 deletions

View File

@ -2776,19 +2776,13 @@ static size_t count_outstanding_requests(chain_head *head)
count += node->lock; count += node->lock;
if (node->dnskey_req && if (!_getdns_netreq_finished(node->dnskey_req))
node->dnskey_req->state != NET_REQ_FINISHED &&
node->dnskey_req->state != NET_REQ_CANCELED)
count++; count++;
if (node->ds_req && if (!_getdns_netreq_finished(node->ds_req))
node->ds_req->state != NET_REQ_FINISHED &&
node->ds_req->state != NET_REQ_CANCELED)
count++; count++;
if (node->soa_req && if (!_getdns_netreq_finished(node->soa_req))
node->soa_req->state != NET_REQ_FINISHED &&
node->soa_req->state != NET_REQ_CANCELED)
count++; count++;
} }
return count + count_outstanding_requests(head->next); return count + count_outstanding_requests(head->next);

View File

@ -104,8 +104,7 @@ _getdns_check_dns_req_complete(getdns_dns_req *dns_req)
int results_found = 0, r; int results_found = 0, r;
for (netreq_p = dns_req->netreqs; (netreq = *netreq_p); netreq_p++) for (netreq_p = dns_req->netreqs; (netreq = *netreq_p); netreq_p++)
if (netreq->state != NET_REQ_FINISHED && if (!_getdns_netreq_finished(netreq))
netreq->state != NET_REQ_CANCELED)
return; return;
else if (netreq->response_len > 0) else if (netreq->response_len > 0)
results_found = 1; results_found = 1;

View File

@ -565,14 +565,12 @@ stub_timeout_cb(void *userarg)
stub_next_upstream(netreq); stub_next_upstream(netreq);
stub_cleanup(netreq); stub_cleanup(netreq);
if (netreq->fd >= 0) close(netreq->fd); if (netreq->fd >= 0) close(netreq->fd);
if (netreq->owner->user_callback) {
netreq->state = NET_REQ_TIMED_OUT; netreq->state = NET_REQ_TIMED_OUT;
if (netreq->owner->user_callback) {
netreq->debug_end_time = _getdns_get_time_as_uintt64(); netreq->debug_end_time = _getdns_get_time_as_uintt64();
(void) _getdns_context_request_timed_out(netreq->owner); (void) _getdns_context_request_timed_out(netreq->owner);
} else { } else
netreq->state = NET_REQ_FINISHED;
_getdns_check_dns_req_complete(netreq->owner); _getdns_check_dns_req_complete(netreq->owner);
}
} }

View File

@ -154,11 +154,12 @@ priv_getdns_context_mf(getdns_context *context);
typedef enum network_req_state_enum typedef enum network_req_state_enum
{ {
NET_REQ_NOT_SENT, NET_REQ_NOT_SENT = 0,
NET_REQ_IN_FLIGHT, NET_REQ_IN_FLIGHT = 1,
NET_REQ_FINISHED, NET_REQ_FINISHED = 2, /* Finish type in bits 2 and 3 */
NET_REQ_CANCELED, NET_REQ_CANCELED = 6, /* 2 + (1 << 2) */
NET_REQ_TIMED_OUT NET_REQ_TIMED_OUT = 10, /* 2 + (2 << 2) */
NET_REQ_ERRORED = 14 /* 2 + (3 << 2) */
} network_req_state; } network_req_state;
@ -257,10 +258,11 @@ typedef struct getdns_network_req
size_t wire_data_sz; size_t wire_data_sz;
uint8_t wire_data[]; uint8_t wire_data[];
} getdns_network_req; } getdns_network_req;
static inline int _getdns_netreq_finished(getdns_network_req *req)
{ return !req || (req->state & NET_REQ_FINISHED); }
/** /**
* dns request - manages a number of network requests and * dns request - manages a number of network requests and
* the initial data passed to getdns_general * the initial data passed to getdns_general

View File

@ -1109,6 +1109,18 @@ _getdns_create_getdns_response(getdns_dns_req *completed_request)
for ( netreq_p = completed_request->netreqs for ( netreq_p = completed_request->netreqs
; (netreq = *netreq_p) ; netreq_p++) { ; (netreq = *netreq_p) ; netreq_p++) {
if (call_reporting && ( netreq->response_len
|| netreq->state == NET_REQ_TIMED_OUT)) {
if (!(netreq_debug =
_getdns_create_call_reporting_dict(context,netreq)))
goto error;
if (_getdns_list_append_this_dict(
call_reporting, netreq_debug))
goto error;
netreq_debug = NULL;
}
if (! netreq->response_len) if (! netreq->response_len)
continue; continue;
@ -1171,43 +1183,10 @@ _getdns_create_getdns_response(getdns_dns_req *completed_request)
getdns_dict_destroy(reply); getdns_dict_destroy(reply);
goto error; goto error;
} }
if (call_reporting) {
if (!(netreq_debug =
_getdns_create_call_reporting_dict(context,netreq)))
goto error;
if (_getdns_list_append_this_dict(
call_reporting, netreq_debug)) {
getdns_dict_destroy(netreq_debug);
goto error;
}
}
if (_getdns_list_append_const_bindata(replies_full, if (_getdns_list_append_const_bindata(replies_full,
netreq->response_len, netreq->response)) netreq->response_len, netreq->response))
goto error; goto error;
} }
if (!nreplies && call_reporting) {
for ( netreq_p = completed_request->netreqs
; (netreq = *netreq_p) ; netreq_p++) {
/* Add call_reporting info for timed out request */
if (netreq->state == NET_REQ_TIMED_OUT) {
if (!(netreq_debug =
_getdns_create_call_reporting_dict(context,netreq)))
goto error;
if (_getdns_list_append_this_dict(
call_reporting, netreq_debug)) {
getdns_dict_destroy(netreq_debug);
goto error;
}
}
}
}
if (_getdns_dict_set_this_list(result, "replies_tree", replies_tree)) if (_getdns_dict_set_this_list(result, "replies_tree", replies_tree))
goto error; goto error;
replies_tree = NULL; replies_tree = NULL;