mirror of https://github.com/getdnsapi/getdns.git
171 lines
4.6 KiB
C
171 lines
4.6 KiB
C
/**
|
|
*
|
|
* /brief getdns contect management functions
|
|
*
|
|
* This is the meat of the API
|
|
* Originally taken from the getdns API description pseudo implementation.
|
|
*
|
|
*/
|
|
/*
|
|
* Copyright (c) 2013, Versign, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* 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.
|
|
* * Neither the name of the <organization> nor the
|
|
* names of its contributors may be used to endorse or promote products
|
|
* derived from this software without specific prior written permission.
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
#include "config.h"
|
|
#include "types-internal.h"
|
|
#include "util-internal.h"
|
|
|
|
void
|
|
network_req_free(getdns_network_req * net_req)
|
|
{
|
|
if (!net_req) {
|
|
return;
|
|
}
|
|
struct getdns_context *context = net_req->owner->context;
|
|
if (net_req->result) {
|
|
ldns_pkt_free(net_req->result);
|
|
}
|
|
GETDNS_FREE(context->mf, net_req);
|
|
}
|
|
|
|
getdns_network_req *
|
|
network_req_new(getdns_dns_req * owner,
|
|
uint16_t request_type,
|
|
uint16_t request_class, struct getdns_dict *extensions)
|
|
{
|
|
|
|
struct getdns_context *context = owner->context;
|
|
getdns_network_req *net_req = GETDNS_MALLOC( context->mf
|
|
, getdns_network_req);
|
|
if (!net_req) {
|
|
return NULL;
|
|
}
|
|
net_req->result = NULL;
|
|
net_req->next = NULL;
|
|
|
|
net_req->request_type = request_type;
|
|
net_req->request_class = request_class;
|
|
net_req->unbound_id = -1;
|
|
net_req->state = NET_REQ_NOT_SENT;
|
|
net_req->owner = owner;
|
|
|
|
/* TODO: records and other extensions */
|
|
|
|
return net_req;
|
|
}
|
|
|
|
void
|
|
dns_req_free(getdns_dns_req * req)
|
|
{
|
|
if (!req) {
|
|
return;
|
|
}
|
|
getdns_network_req *net_req = NULL;
|
|
struct getdns_context *context = req->context;
|
|
|
|
/* free extensions */
|
|
getdns_dict_destroy(req->extensions);
|
|
|
|
/* free network requests */
|
|
net_req = req->first_req;
|
|
while (net_req) {
|
|
getdns_network_req *next = net_req->next;
|
|
network_req_free(net_req);
|
|
net_req = next;
|
|
}
|
|
|
|
if (req->local_timeout_id != 0) {
|
|
getdns_context_clear_timeout(context, req->local_timeout_id);
|
|
}
|
|
|
|
getdns_context_clear_timeout(context, req->trans_id);
|
|
|
|
/* free strduped name */
|
|
free(req->name);
|
|
|
|
GETDNS_FREE(context->mf, req);
|
|
}
|
|
|
|
/* create a new dns req to be submitted */
|
|
getdns_dns_req *
|
|
dns_req_new(struct getdns_context *context,
|
|
const char *name, uint16_t request_type, struct getdns_dict *extensions)
|
|
{
|
|
|
|
getdns_dns_req *result = NULL;
|
|
getdns_network_req *req = NULL;
|
|
|
|
result = GETDNS_MALLOC(context->mf, getdns_dns_req);
|
|
if (result == NULL) {
|
|
return NULL;
|
|
}
|
|
|
|
result->name = strdup(name);
|
|
result->context = context;
|
|
result->canceled = 0;
|
|
result->current_req = NULL;
|
|
result->first_req = NULL;
|
|
result->trans_id = ldns_get_random();
|
|
|
|
getdns_dict_copy(extensions, &result->extensions);
|
|
|
|
/* will be set by caller */
|
|
result->user_pointer = NULL;
|
|
result->user_callback = NULL;
|
|
result->local_timeout_id = 0;
|
|
|
|
/* create the requests */
|
|
req = network_req_new(result,
|
|
request_type, LDNS_RR_CLASS_IN, extensions);
|
|
if (!req) {
|
|
dns_req_free(result);
|
|
return NULL;
|
|
}
|
|
|
|
result->current_req = req;
|
|
result->first_req = req;
|
|
|
|
/* tack on A or AAAA if needed */
|
|
if (is_extension_set(extensions, "return_both_v4_and_v6") &&
|
|
(request_type == GETDNS_RRTYPE_A ||
|
|
request_type == GETDNS_RRTYPE_AAAA)) {
|
|
|
|
uint16_t next_req_type =
|
|
(request_type ==
|
|
GETDNS_RRTYPE_A) ? GETDNS_RRTYPE_AAAA : GETDNS_RRTYPE_A;
|
|
getdns_network_req *next_req = network_req_new(result,
|
|
next_req_type,
|
|
LDNS_RR_CLASS_IN,
|
|
extensions);
|
|
if (!next_req) {
|
|
dns_req_free(result);
|
|
return NULL;
|
|
}
|
|
req->next = next_req;
|
|
}
|
|
|
|
return result;
|
|
}
|