2013-07-02 16:00:20 -05:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* /brief getdns core functions for synchronous use
|
2013-10-17 18:45:25 -05:00
|
|
|
*
|
2013-07-02 16:00:20 -05:00
|
|
|
* Originally taken from the getdns API description pseudo implementation.
|
|
|
|
*
|
|
|
|
*/
|
2013-11-04 17:37:54 -06:00
|
|
|
|
|
|
|
/*
|
2014-02-25 07:12:33 -06:00
|
|
|
* Copyright (c) 2013, NLnet Labs, Verisign, Inc.
|
2013-11-04 17:37:54 -06:00
|
|
|
* All rights reserved.
|
2013-12-09 11:55:33 -06:00
|
|
|
*
|
2013-11-04 17:37:54 -06:00
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions are met:
|
|
|
|
* * Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* * Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
2014-02-25 07:23:19 -06:00
|
|
|
* * Neither the names of the copyright holders nor the
|
2013-11-04 17:37:54 -06:00
|
|
|
* names of its contributors may be used to endorse or promote products
|
|
|
|
* derived from this software without specific prior written permission.
|
2013-10-17 18:45:25 -05:00
|
|
|
*
|
2013-11-04 17:37:54 -06:00
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
|
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
|
|
* DISCLAIMED. IN NO EVENT SHALL Verisign, Inc. BE LIABLE FOR ANY
|
|
|
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
|
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
|
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
|
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
|
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
2013-07-02 16:00:20 -05:00
|
|
|
*/
|
|
|
|
|
2014-05-19 08:50:34 -05:00
|
|
|
#include <string.h>
|
2014-01-21 14:31:22 -06:00
|
|
|
#include <unbound.h>
|
2014-05-19 08:50:34 -05:00
|
|
|
#include "getdns/getdns.h"
|
|
|
|
#include "config.h"
|
2013-10-15 16:28:23 -05:00
|
|
|
#include "context.h"
|
|
|
|
#include "general.h"
|
2013-12-03 07:13:10 -06:00
|
|
|
#include "types-internal.h"
|
2013-10-31 14:08:23 -05:00
|
|
|
#include "util-internal.h"
|
2014-02-19 09:56:00 -06:00
|
|
|
#include "dnssec.h"
|
2014-07-01 16:31:40 -05:00
|
|
|
#include "ub_timed_resolve.h"
|
2013-07-02 16:00:20 -05:00
|
|
|
|
2014-09-16 08:43:20 -05:00
|
|
|
#include "stub.h"
|
|
|
|
#include "gldns/wire2str.h"
|
|
|
|
|
2013-07-02 16:00:20 -05:00
|
|
|
/* stuff to make it compile pedantically */
|
|
|
|
#define UNUSED_PARAM(x) ((void)(x))
|
2013-12-09 11:55:33 -06:00
|
|
|
#define RETURN_IF_NULL(ptr, code) if(ptr == NULL) return code;
|
2013-07-02 16:00:20 -05:00
|
|
|
|
2014-09-12 07:44:53 -05:00
|
|
|
static getdns_return_t submit_request_sync_rec(
|
2014-07-01 16:31:40 -05:00
|
|
|
getdns_dns_req* req, uint64_t *timeout)
|
|
|
|
{
|
2014-01-21 14:31:22 -06:00
|
|
|
struct ub_result* ub_res = NULL;
|
|
|
|
getdns_return_t gr = GETDNS_RETURN_GOOD;
|
|
|
|
getdns_network_req *netreq = req->first_req;
|
2014-07-01 16:31:40 -05:00
|
|
|
|
2014-01-21 14:31:22 -06:00
|
|
|
while (netreq) {
|
2014-07-01 16:31:40 -05:00
|
|
|
int r = ub_timed_resolve(req->context->unbound_ctx,
|
2014-01-21 14:31:22 -06:00
|
|
|
req->name,
|
|
|
|
netreq->request_type,
|
|
|
|
netreq->request_class,
|
2014-07-01 16:31:40 -05:00
|
|
|
&ub_res,
|
|
|
|
timeout);
|
2014-01-21 14:31:22 -06:00
|
|
|
gr = getdns_apply_network_result(netreq, ub_res);
|
|
|
|
ub_resolve_free(ub_res);
|
|
|
|
ub_res = NULL;
|
2014-07-01 16:31:40 -05:00
|
|
|
|
|
|
|
if (r != GETDNS_RETURN_GOOD)
|
|
|
|
return r;
|
|
|
|
else if (gr != GETDNS_RETURN_GOOD)
|
2014-01-21 14:31:22 -06:00
|
|
|
return gr;
|
2014-07-01 16:31:40 -05:00
|
|
|
|
2014-01-21 14:31:22 -06:00
|
|
|
netreq = netreq->next;
|
|
|
|
}
|
|
|
|
return gr;
|
2013-08-16 15:28:21 -05:00
|
|
|
}
|
|
|
|
|
2014-09-16 08:43:20 -05:00
|
|
|
static getdns_return_t submit_request_sync_stub(
|
|
|
|
getdns_dns_req* req, uint64_t *timeout)
|
|
|
|
{
|
2014-09-23 03:44:48 -05:00
|
|
|
ldns_rdf *qname;
|
|
|
|
getdns_network_req *netreq = req->first_req;
|
|
|
|
uint16_t qflags = 0;
|
2014-09-12 07:44:53 -05:00
|
|
|
struct timeval tv;
|
2014-09-26 18:07:42 -05:00
|
|
|
gldns_buffer *gbuffer = gldns_buffer_new(4096);
|
2014-09-12 07:44:53 -05:00
|
|
|
|
2014-09-23 03:44:48 -05:00
|
|
|
while (netreq) {
|
2014-09-26 18:07:42 -05:00
|
|
|
gldns_buffer_clear(gbuffer);
|
|
|
|
(void) getdns_stub_dns_query_sync(req->context, req->name,
|
|
|
|
netreq->request_type, req->extensions, gbuffer);
|
2014-09-12 07:44:53 -05:00
|
|
|
|
2014-09-23 03:44:48 -05:00
|
|
|
qname = ldns_dname_new_frm_str(req->name);
|
|
|
|
qflags = qflags | LDNS_RD;
|
|
|
|
/* TODO: Use timeout properly - create a ldns_timed_resolve function */
|
|
|
|
/* timeout is in miliseconds, so map to seconds and microseconds */
|
|
|
|
tv.tv_sec = *timeout / 1000;
|
|
|
|
tv.tv_usec = (*timeout % 1000) * 1000;
|
|
|
|
ldns_resolver_set_timeout(req->context->ldns_res, tv);
|
|
|
|
netreq->result = ldns_resolver_query(
|
2014-09-26 15:17:42 -05:00
|
|
|
req->context->ldns_res, qname, netreq->request_type,
|
|
|
|
netreq->request_class, qflags);
|
|
|
|
/*TODO: The rec unbound case always sends DO=1 and then
|
|
|
|
getdns_apply_network_result sets these values...*/
|
|
|
|
// netreq->secure = ;
|
|
|
|
// netreq->bogus = ;
|
2014-09-23 03:44:48 -05:00
|
|
|
ldns_rdf_deep_free(qname);
|
|
|
|
qname = NULL;
|
2014-09-12 07:44:53 -05:00
|
|
|
|
2014-09-23 03:44:48 -05:00
|
|
|
if (! netreq->result) {
|
|
|
|
/* TODO: use better errors */
|
|
|
|
return GETDNS_RETURN_GENERIC_ERROR;
|
|
|
|
}
|
2014-09-26 15:17:42 -05:00
|
|
|
|
2014-09-23 03:44:48 -05:00
|
|
|
netreq = netreq->next;
|
2014-09-16 08:43:20 -05:00
|
|
|
}
|
2014-09-23 03:44:48 -05:00
|
|
|
return GETDNS_RETURN_GOOD;
|
2014-09-16 08:43:20 -05:00
|
|
|
}
|
|
|
|
|
2014-09-12 07:44:53 -05:00
|
|
|
static getdns_return_t submit_request_sync(
|
|
|
|
getdns_dns_req* req, struct getdns_context *context)
|
|
|
|
{
|
2014-10-13 08:36:21 -05:00
|
|
|
uint64_t timeout = context->timeout;
|
|
|
|
if (context->resolution_type == GETDNS_RESOLUTION_STUB) {
|
|
|
|
return submit_request_sync_stub(req, &timeout);
|
|
|
|
} else {
|
|
|
|
return submit_request_sync_rec(req, &timeout);
|
|
|
|
}
|
2014-09-12 07:44:53 -05:00
|
|
|
}
|
2014-09-16 08:43:20 -05:00
|
|
|
|
2014-09-23 06:36:26 -05:00
|
|
|
getdns_return_t
|
|
|
|
getdns_general_sync_ns(struct getdns_context *context,
|
|
|
|
const char *name,
|
|
|
|
uint16_t request_type,
|
|
|
|
struct getdns_dict *extensions,
|
|
|
|
struct getdns_dict **response,
|
|
|
|
bool usenamespaces)
|
|
|
|
{
|
|
|
|
getdns_dns_req *req;
|
|
|
|
getdns_return_t response_status;
|
|
|
|
uint64_t timeout;
|
|
|
|
|
|
|
|
RETURN_IF_NULL(context, GETDNS_RETURN_INVALID_PARAMETER);
|
|
|
|
RETURN_IF_NULL(response, GETDNS_RETURN_INVALID_PARAMETER);
|
|
|
|
RETURN_IF_NULL(name, GETDNS_RETURN_INVALID_PARAMETER);
|
|
|
|
|
|
|
|
timeout = context->timeout;
|
|
|
|
response_status = validate_dname(name);
|
|
|
|
if (response_status != GETDNS_RETURN_GOOD)
|
|
|
|
return response_status;
|
|
|
|
|
|
|
|
response_status = validate_extensions(extensions);
|
|
|
|
if (response_status != GETDNS_RETURN_GOOD)
|
|
|
|
return response_status;
|
|
|
|
|
|
|
|
/* Set up the context assuming we won't use the specified namespaces.
|
|
|
|
This is (currently) identical to setting up a pure DNS namespace */
|
|
|
|
response_status = getdns_context_prepare_for_resolution(context, 0);
|
|
|
|
if (response_status != GETDNS_RETURN_GOOD)
|
|
|
|
return response_status;
|
|
|
|
|
|
|
|
/* create the request */
|
2014-10-13 17:14:25 -05:00
|
|
|
req = dns_req_new(context, context->extension, name, request_type, extensions);
|
2014-09-23 06:36:26 -05:00
|
|
|
if (!req)
|
|
|
|
return GETDNS_RETURN_MEMORY_ERROR;
|
|
|
|
|
|
|
|
/* resolve using the appropriate namespace*/
|
|
|
|
if (!usenamespaces) {
|
|
|
|
response_status = submit_request_sync(req, context);
|
|
|
|
} else {
|
|
|
|
for (int i = 0; i < context->namespace_count; i++) {
|
|
|
|
switch (context->namespaces[i]) {
|
|
|
|
case GETDNS_NAMESPACE_LOCALNAMES:
|
2014-09-25 09:25:22 -05:00
|
|
|
response_status = getdns_context_local_namespace_resolve(req,
|
|
|
|
response,
|
|
|
|
context);
|
|
|
|
/* For a local lookup the response is populated directly*/
|
|
|
|
if (response_status == GETDNS_RETURN_GOOD) {
|
|
|
|
dns_req_free(req);
|
|
|
|
return response_status;
|
|
|
|
}
|
2014-09-23 06:36:26 -05:00
|
|
|
break;
|
|
|
|
|
|
|
|
case GETDNS_NAMESPACE_DNS:
|
2014-09-25 09:25:22 -05:00
|
|
|
/* TODO: We will get a good return code here even if the name is
|
|
|
|
not found (NXDOMAIN). We should consider if this means we
|
|
|
|
go onto the next namespace instead of returning*/
|
2014-09-23 06:36:26 -05:00
|
|
|
response_status = submit_request_sync(req, context);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
response_status = GETDNS_RETURN_BAD_CONTEXT;
|
|
|
|
break;
|
|
|
|
}
|
2014-09-25 09:25:22 -05:00
|
|
|
/* If we have a good response break out the for loop as we are done,
|
2014-09-23 06:36:26 -05:00
|
|
|
but if we don't then give the next namespace a try*/
|
|
|
|
if (response_status == GETDNS_RETURN_GOOD)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-25 09:25:22 -05:00
|
|
|
/* Only get here if the response came from the DNS namespace*/
|
2014-09-23 06:36:26 -05:00
|
|
|
if (response_status == GETDNS_RETURN_GOOD) {
|
|
|
|
if (is_extension_set(req->extensions,
|
|
|
|
"dnssec_return_validation_chain"))
|
|
|
|
*response = priv_getdns_get_validation_chain_sync(req, &timeout);
|
|
|
|
else
|
|
|
|
*response = create_getdns_response(req);
|
|
|
|
|
|
|
|
} else if (response_status == GETDNS_RESPSTATUS_ALL_TIMEOUT) {
|
|
|
|
*response = create_getdns_response(req);
|
|
|
|
response_status = GETDNS_RETURN_GOOD;
|
|
|
|
}
|
|
|
|
|
|
|
|
dns_req_free(req);
|
|
|
|
return response_status;
|
|
|
|
}
|
|
|
|
|
2013-07-02 16:00:20 -05:00
|
|
|
getdns_return_t
|
2013-12-06 08:54:06 -06:00
|
|
|
getdns_general_sync(struct getdns_context *context,
|
2013-11-05 14:03:44 -06:00
|
|
|
const char *name,
|
|
|
|
uint16_t request_type,
|
|
|
|
struct getdns_dict *extensions,
|
2013-11-08 20:39:06 -06:00
|
|
|
struct getdns_dict **response)
|
2013-08-16 15:28:21 -05:00
|
|
|
{
|
2014-09-23 06:36:26 -05:00
|
|
|
/* general, so without dns lookup (no namespaces) */;
|
|
|
|
return getdns_general_sync_ns(context, name, request_type,
|
|
|
|
extensions, response, false);
|
2013-08-16 15:28:21 -05:00
|
|
|
}
|
2013-07-02 16:00:20 -05:00
|
|
|
|
|
|
|
getdns_return_t
|
2013-12-06 08:54:06 -06:00
|
|
|
getdns_address_sync(struct getdns_context *context,
|
2013-11-05 14:03:44 -06:00
|
|
|
const char *name,
|
|
|
|
struct getdns_dict * extensions,
|
2013-11-08 20:39:06 -06:00
|
|
|
struct getdns_dict ** response)
|
2013-08-16 15:28:21 -05:00
|
|
|
{
|
2013-11-05 14:03:44 -06:00
|
|
|
int cleanup_extensions = 0;
|
|
|
|
if (!extensions) {
|
2013-11-11 16:10:22 -06:00
|
|
|
extensions = getdns_dict_create_with_context(context);
|
2013-11-05 14:03:44 -06:00
|
|
|
cleanup_extensions = 1;
|
|
|
|
}
|
|
|
|
getdns_dict_set_int(extensions,
|
|
|
|
GETDNS_STR_EXTENSION_RETURN_BOTH_V4_AND_V6, GETDNS_EXTENSION_TRUE);
|
|
|
|
|
|
|
|
getdns_return_t result =
|
2014-09-23 06:36:26 -05:00
|
|
|
getdns_general_sync_ns(context, name, GETDNS_RRTYPE_A,
|
|
|
|
extensions, response, true);
|
2013-11-05 14:03:44 -06:00
|
|
|
if (cleanup_extensions) {
|
|
|
|
getdns_dict_destroy(extensions);
|
|
|
|
}
|
|
|
|
return result;
|
2013-08-16 15:28:21 -05:00
|
|
|
}
|
2013-07-02 16:00:20 -05:00
|
|
|
|
|
|
|
getdns_return_t
|
2013-12-06 08:54:06 -06:00
|
|
|
getdns_hostname_sync(struct getdns_context *context,
|
2013-11-05 14:03:44 -06:00
|
|
|
struct getdns_dict * address,
|
|
|
|
struct getdns_dict * extensions,
|
2013-11-08 20:39:06 -06:00
|
|
|
struct getdns_dict ** response)
|
2013-10-30 15:16:20 -05:00
|
|
|
{
|
2013-11-05 14:03:44 -06:00
|
|
|
struct getdns_bindata *address_data;
|
|
|
|
struct getdns_bindata *address_type;
|
|
|
|
uint16_t req_type;
|
|
|
|
char *name;
|
|
|
|
getdns_return_t retval;
|
|
|
|
|
|
|
|
if ((retval =
|
|
|
|
getdns_dict_get_bindata(address, "address_data",
|
|
|
|
&address_data)) != GETDNS_RETURN_GOOD)
|
|
|
|
return retval;
|
|
|
|
if ((retval =
|
|
|
|
getdns_dict_get_bindata(address, "address_type",
|
|
|
|
&address_type)) != GETDNS_RETURN_GOOD)
|
|
|
|
return retval;
|
|
|
|
if ((strncmp(GETDNS_STR_IPV4, (char *) address_type->data,
|
2013-12-11 16:41:21 -06:00
|
|
|
( strlen(GETDNS_STR_IPV4) < address_type->size
|
|
|
|
? strlen(GETDNS_STR_IPV4) : address_type->size )) == 0
|
|
|
|
&& address_data->size == 4)
|
2013-11-05 14:03:44 -06:00
|
|
|
|| (strncmp(GETDNS_STR_IPV6, (char *) address_type->data,
|
2013-12-11 16:41:21 -06:00
|
|
|
( strlen(GETDNS_STR_IPV6) < address_type->size
|
|
|
|
? strlen(GETDNS_STR_IPV6) : address_type->size )) == 0
|
|
|
|
&& address_data->size == 16))
|
2013-11-05 14:03:44 -06:00
|
|
|
req_type = GETDNS_RRTYPE_PTR;
|
|
|
|
else
|
2014-02-03 20:34:11 -06:00
|
|
|
return GETDNS_RETURN_INVALID_PARAMETER;
|
2013-12-11 16:41:21 -06:00
|
|
|
if ((name = reverse_address(address_data)) == NULL)
|
2014-02-03 20:34:11 -06:00
|
|
|
return GETDNS_RETURN_INVALID_PARAMETER;
|
2014-09-23 06:36:26 -05:00
|
|
|
retval = getdns_general_sync_ns(context, name, req_type, extensions,
|
|
|
|
response, true);
|
2013-12-11 16:41:21 -06:00
|
|
|
free(name);
|
|
|
|
return retval;
|
2013-10-30 15:16:20 -05:00
|
|
|
}
|
|
|
|
|
2013-07-02 16:00:20 -05:00
|
|
|
getdns_return_t
|
2013-12-06 08:54:06 -06:00
|
|
|
getdns_service_sync(struct getdns_context *context,
|
2013-11-05 14:03:44 -06:00
|
|
|
const char *name,
|
|
|
|
struct getdns_dict * extensions,
|
2013-11-08 20:39:06 -06:00
|
|
|
struct getdns_dict ** response)
|
2013-08-16 15:28:21 -05:00
|
|
|
{
|
2013-10-17 18:45:25 -05:00
|
|
|
|
2014-09-23 06:36:26 -05:00
|
|
|
return getdns_general_sync_ns(context, name, GETDNS_RRTYPE_SRV,
|
|
|
|
extensions, response, true);
|
2013-08-16 15:28:21 -05:00
|
|
|
|
|
|
|
}
|
2013-07-02 16:00:20 -05:00
|
|
|
|
|
|
|
void
|
2013-11-05 14:03:44 -06:00
|
|
|
getdns_free_sync_request_memory(struct getdns_dict *response)
|
2013-11-04 11:51:13 -06:00
|
|
|
{
|
2013-11-05 14:03:44 -06:00
|
|
|
getdns_dict_destroy(response);
|
2013-11-04 11:51:13 -06:00
|
|
|
}
|
2013-11-05 14:03:44 -06:00
|
|
|
|
2013-07-02 16:00:20 -05:00
|
|
|
/* getdns_core_sync.c */
|