Fix canonical_name at root of response dict

It will be the first canonical name in the replies.
This commit is contained in:
Willem Toorop 2015-02-18 14:46:55 +01:00
parent a77f156d08
commit f78fdd0594
1 changed files with 82 additions and 130 deletions

View File

@ -176,16 +176,6 @@ sockaddr_to_dict(struct getdns_context *context, struct sockaddr_storage *addres
return GETDNS_RETURN_GOOD; return GETDNS_RETURN_GOOD;
} }
/* result must be freed */
static char *
convert_rdf_to_str(ldns_rdf * rdf)
{
if (ldns_rdf_get_type(rdf) == LDNS_RDF_TYPE_DNAME) {
ldns_dname2canonical(rdf);
}
return ldns_rdf2str(rdf);
}
#define SET_WIRE_INT(X,Y) if (getdns_dict_set_int(result, #X , (int) \ #define SET_WIRE_INT(X,Y) if (getdns_dict_set_int(result, #X , (int) \
GLDNS_ ## Y ## _WIRE(netreq->response))) break GLDNS_ ## Y ## _WIRE(netreq->response))) break
#define SET_WIRE_CNT(X,Y) if (getdns_dict_set_int(result, #X , (int) \ #define SET_WIRE_CNT(X,Y) if (getdns_dict_set_int(result, #X , (int) \
@ -761,36 +751,29 @@ error:
return NULL; return NULL;
} }
static char * getdns_dict *
get_canonical_name(const char *name)
{
ldns_rdf *rdf = ldns_rdf_new_frm_str(LDNS_RDF_TYPE_DNAME, name);
if (!rdf) {
return NULL;
}
char *result = convert_rdf_to_str(rdf);
ldns_rdf_deep_free(rdf);
return result;
}
struct getdns_dict *
create_getdns_response(getdns_dns_req *completed_request) create_getdns_response(getdns_dns_req *completed_request)
{ {
getdns_dict *result = getdns_dict_create_with_context( getdns_dict *result;
completed_request->context);
getdns_list *replies_full = getdns_list_create_with_context(
completed_request->context);
getdns_list *just_addrs = NULL; getdns_list *just_addrs = NULL;
getdns_list *replies_tree = getdns_list_create_with_context( getdns_list *replies_full;
completed_request->context); getdns_list *replies_tree;
getdns_network_req *netreq, **netreq_p; getdns_network_req *netreq, **netreq_p;
char *canonical_name = NULL; int rrsigs_in_answer = 0;
getdns_return_t r = 0; getdns_dict *reply;
getdns_bindata *canonical_name = NULL;
int nreplies = 0, nanswers = 0, nsecure = 0, ninsecure = 0, nbogus = 0; int nreplies = 0, nanswers = 0, nsecure = 0, ninsecure = 0, nbogus = 0;
struct getdns_bindata full_data; getdns_bindata full_data;
/* info (bools) about dns_req */ /* info (bools) about dns_req */
int dnssec_return_status; int dnssec_return_status;
getdns_context *context;
assert(completed_request);
context = completed_request->context;
if (!(result = getdns_dict_create_with_context(context)))
return NULL;
dnssec_return_status = completed_request->dnssec_return_status || dnssec_return_status = completed_request->dnssec_return_status ||
completed_request->dnssec_return_only_secure; completed_request->dnssec_return_only_secure;
@ -800,24 +783,18 @@ create_getdns_response(getdns_dns_req *completed_request)
just_addrs = getdns_list_create_with_context( just_addrs = getdns_list_create_with_context(
completed_request->context); completed_request->context);
do { if (getdns_dict_set_int(result, GETDNS_STR_KEY_ANSWER_TYPE,
canonical_name = get_canonical_name(completed_request->name); GETDNS_NAMETYPE_DNS))
r = getdns_dict_util_set_string(result, GETDNS_STR_KEY_CANONICAL_NM, goto error_free_result;
canonical_name);
free(canonical_name);
if (r != GETDNS_RETURN_GOOD) {
break;
}
r = getdns_dict_set_int(result, GETDNS_STR_KEY_ANSWER_TYPE, if (!(replies_full = getdns_list_create_with_context(context)))
GETDNS_NAMETYPE_DNS); goto error_free_result;
if (r != GETDNS_RETURN_GOOD) {
break; if (!(replies_tree = getdns_list_create_with_context(context)))
} goto error_free_replies_full;
for ( netreq_p = completed_request->netreqs for ( netreq_p = completed_request->netreqs
; ! r && (netreq = *netreq_p) ; (netreq = *netreq_p) ; netreq_p++) {
; netreq_p++) {
if (! netreq->response_len) if (! netreq->response_len)
continue; continue;
@ -836,104 +813,79 @@ create_getdns_response(getdns_dns_req *completed_request)
if (! completed_request->dnssec_return_validation_chain) { if (! completed_request->dnssec_return_validation_chain) {
if (dnssec_return_status && netreq->bogus) if (dnssec_return_status && netreq->bogus)
continue; continue;
else if (completed_request->dnssec_return_only_secure && ! netreq->secure) else if (completed_request->dnssec_return_only_secure
&& ! netreq->secure)
continue; continue;
} }
size_t idx = 0; if (!(reply = create_reply_dict(context,
/* reply tree */ netreq, just_addrs, &rrsigs_in_answer)))
int rrsigs_in_answer = 0; goto error;
struct getdns_dict *reply = create_reply_dict(
completed_request->context, netreq, just_addrs, &rrsigs_in_answer);
if (! reply) { if (!canonical_name) {
r = GETDNS_RETURN_MEMORY_ERROR; if (getdns_dict_get_bindata(
break; reply, "canonical_name", &canonical_name))
goto error;
if (getdns_dict_set_bindata(
result, "canonical_name", canonical_name))
goto error;
} }
if (dnssec_return_status || completed_request->dnssec_return_validation_chain) { if (dnssec_return_status ||
r = getdns_dict_set_int(reply, "dnssec_status", completed_request->dnssec_return_validation_chain) {
if (getdns_dict_set_int(reply, "dnssec_status",
( netreq->secure ? GETDNS_DNSSEC_SECURE ( netreq->secure ? GETDNS_DNSSEC_SECURE
: netreq->bogus ? GETDNS_DNSSEC_BOGUS : netreq->bogus ? GETDNS_DNSSEC_BOGUS
: rrsigs_in_answer && : rrsigs_in_answer &&
completed_request->context->has_ta context->has_ta ? GETDNS_DNSSEC_INDETERMINATE
? GETDNS_DNSSEC_INDETERMINATE : GETDNS_DNSSEC_INSECURE )))
: GETDNS_DNSSEC_INSECURE )); goto error;
if (r != GETDNS_RETURN_GOOD) {
getdns_dict_destroy(reply);
break;
}
}
r = getdns_list_add_item(replies_tree, &idx);
if (r != GETDNS_RETURN_GOOD) {
getdns_dict_destroy(reply);
// break inner while
break;
} }
r = getdns_list_set_dict(replies_tree, idx, reply); if (getdns_list_append_dict(replies_tree, reply)) {
getdns_dict_destroy(reply); getdns_dict_destroy(reply);
if (r != GETDNS_RETURN_GOOD) { goto error;
// break inner while
break;
} }
getdns_dict_destroy(reply);
/* buffer */ /* buffer */
r = getdns_list_add_item(replies_full, &idx);
if (r != GETDNS_RETURN_GOOD) {
// break inner while
break;
}
full_data.data = netreq->response; full_data.data = netreq->response;
full_data.size = netreq->response_len; full_data.size = netreq->response_len;
r = getdns_list_set_bindata(replies_full, idx, if (getdns_list_append_bindata(replies_full, &full_data))
&full_data); goto error;
if (r != GETDNS_RETURN_GOOD) {
free(full_data.data);
// break inner while
break;
}
} }
if (getdns_dict_set_list(result, "replies_tree", replies_tree))
goto error;
getdns_list_destroy(replies_tree);
if (r != GETDNS_RETURN_GOOD) if (getdns_dict_set_list(result, "replies_full", replies_full))
break; goto error_free_replies_full;
getdns_list_destroy(replies_full);
r = getdns_dict_set_list(result, GETDNS_STR_KEY_REPLIES_TREE, if (just_addrs && getdns_dict_set_list(
replies_tree); result, GETDNS_STR_KEY_JUST_ADDRS, just_addrs))
if (r != GETDNS_RETURN_GOOD) goto error_free_result;
break; getdns_list_destroy(just_addrs);
r = getdns_dict_set_list(result, GETDNS_STR_KEY_REPLIES_FULL, if (getdns_dict_set_int(result, GETDNS_STR_KEY_STATUS,
replies_full);
if (r != GETDNS_RETURN_GOOD)
break;
if (just_addrs) {
r = getdns_dict_set_list(result, GETDNS_STR_KEY_JUST_ADDRS,
just_addrs);
if (r != GETDNS_RETURN_GOOD) {
break;
}
}
r = getdns_dict_set_int(result, GETDNS_STR_KEY_STATUS,
nreplies == 0 ? GETDNS_RESPSTATUS_ALL_TIMEOUT : nreplies == 0 ? GETDNS_RESPSTATUS_ALL_TIMEOUT :
completed_request->dnssec_return_only_secure && nsecure == 0 && ninsecure > 0 completed_request->dnssec_return_only_secure && nsecure == 0 && ninsecure > 0
? GETDNS_RESPSTATUS_NO_SECURE_ANSWERS : ? GETDNS_RESPSTATUS_NO_SECURE_ANSWERS :
completed_request->dnssec_return_only_secure && nsecure == 0 && nbogus > 0 completed_request->dnssec_return_only_secure && nsecure == 0 && nbogus > 0
? GETDNS_RESPSTATUS_ALL_BOGUS_ANSWERS : ? GETDNS_RESPSTATUS_ALL_BOGUS_ANSWERS :
nanswers == 0 ? GETDNS_RESPSTATUS_NO_NAME nanswers == 0 ? GETDNS_RESPSTATUS_NO_NAME
: GETDNS_RESPSTATUS_GOOD); : GETDNS_RESPSTATUS_GOOD))
} while (0); goto error_free_result;
/* cleanup */
getdns_list_destroy(replies_tree);
getdns_list_destroy(replies_full);
getdns_list_destroy(just_addrs);
if (r != 0) {
getdns_dict_destroy(result);
result = NULL;
}
return result; return result;
error:
/* cleanup */
getdns_list_destroy(replies_tree);
error_free_replies_full:
getdns_list_destroy(replies_full);
error_free_result:
getdns_list_destroy(just_addrs);
getdns_dict_destroy(result);
return NULL;
} }
/*This method can be used when e.g. a local lookup has been performed and the /*This method can be used when e.g. a local lookup has been performed and the
@ -948,7 +900,8 @@ create_getdns_response_from_rr_list(struct getdns_dns_req * completed_request,
struct getdns_list *replies_tree = getdns_list_create_with_context( struct getdns_list *replies_tree = getdns_list_create_with_context(
completed_request->context); completed_request->context);
struct getdns_list *just_addrs = NULL; struct getdns_list *just_addrs = NULL;
char *canonical_name = NULL; uint8_t canonical_name_space[256];
getdns_bindata bindata = { 256, canonical_name_space };
getdns_return_t r = 0; getdns_return_t r = 0;
/* NOTE: With DNS packet, we ignore any DNSSEC related extensions since we /* NOTE: With DNS packet, we ignore any DNSSEC related extensions since we
@ -957,13 +910,12 @@ create_getdns_response_from_rr_list(struct getdns_dns_req * completed_request,
just_addrs = getdns_list_create_with_context(completed_request->context); just_addrs = getdns_list_create_with_context(completed_request->context);
do { do {
canonical_name = get_canonical_name(completed_request->name); if ((r = gldns_str2wire_dname_buf(completed_request->name,
r = getdns_dict_util_set_string(result, GETDNS_STR_KEY_CANONICAL_NM, bindata.data, &bindata.size)))
canonical_name); break;
free(canonical_name); if ((r = getdns_dict_set_bindata(result,
if (r != GETDNS_RETURN_GOOD) { GETDNS_STR_KEY_CANONICAL_NM, &bindata)))
break; break;
}
/* For local lookups we don't set an answer_type as there isn't a /* For local lookups we don't set an answer_type as there isn't a
suitable one*/ suitable one*/