nsupdate.c revision 15a44745412679c30a6d022733925af70a38b715
/*
* Copyright (C) 2000 Internet Software Consortium.
*
* Permission to use, copy, modify, and distribute this software for any
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice appear in all copies.
*
* THE SOFTWARE IS PROVIDED "AS IS" AND INTERNET SOFTWARE CONSORTIUM
* DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL
* INTERNET SOFTWARE CONSORTIUM BE LIABLE FOR ANY SPECIAL, DIRECT,
* INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING
* FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT,
* NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION
* WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*/
/* $Id: nsupdate.c,v 1.34 2000/07/27 09:37:34 tale Exp $ */
#include <config.h>
#include <isc/condition.h>
#include <isc/commandline.h>
#include <isc/sockaddr.h>
#include <dns/callbacks.h>
#include <dns/dispatch.h>
#include <dns/rdataclass.h>
#include <dns/rdatalist.h>
#include <dns/rdataset.h>
#include <dns/rdatastruct.h>
#include <dns/rdatatype.h>
#include <ctype.h>
#include <netdb.h>
#include <stdlib.h>
#include <unistd.h>
#define MXNAME 256
#define MAXPNAME 1025
#define MAXCMD 1024
#define NAMEBUF 512
#define WORDLEN 512
#define PACKETSIZE 2048
#define MSGTEXT 4096
#define FIND_TIMEOUT 5
#define DNSDEFAULTPORT 53
#define RESOLV_CONF "/etc/resolv.conf"
static isc_mutex_t lock;
static isc_condition_t cond;
static dns_fixedname_t fuserzone;
static lwres_conf_t *lwconf;
static isc_sockaddr_t *servers;
static int ns_inuse = 0;
static int ns_total = 0;
typedef struct nsu_requestinfo {
dns_request_t **request);
#define STATUS_MORE 0
#define STATUS_SEND 1
#define STATUS_QUIT 2
#define STATUS_SYNTAX 3
static void
exit(1);
}
static void
if (debugging) {
}
}
static void
if (ddebugging) {
}
}
static inline void
if (result != ISC_R_SUCCESS)
}
static void *
}
static void
}
static char *
char *s;
const char *d;
return (NULL);
break;
}
if (dc == 0)
break;
}
for (s = string; *s != '\0'; s++) {
sc = *s;
*s++ = '\0';
*stringp = s;
return (string);
}
}
}
return (string);
}
static unsigned int
int i = 0;
while (*s != 0) {
if (*s++ == '.') {
i++;
*last_was_dot = ISC_TRUE;
} else
}
return (i);
}
static void
reset_system(void) {
ddebug("reset_system()");
/* If the update message is still around, destroy it */
else {
&updatemsg);
}
}
static void
setup_key() {
int secretlen;
char *secretstr;
char *s;
debug("Creating key...");
fatal("key option must specify keyname:secret\n");
secretstr = s + 1;
debug("namefromtext");
fatal("out of memory");
if (result != ISC_R_SUCCESS) {
goto failure;
}
debug("close");
} else {
if (result != ISC_R_SUCCESS) {
goto failure;
}
fatal("out of memory");
if (result != ISC_R_SUCCESS) {
goto failure;
}
}
debug("keycreate");
if (result != ISC_R_SUCCESS) {
char *str;
else
}
return;
}
static void
setup_system(void) {
int i;
ddebug("setup_system()");
result = isc_net_probeipv4();
/* XXXMWS There isn't any actual V6 support in the code yet */
result = isc_net_probeipv6();
if (result == ISC_R_SUCCESS)
if (lwresult != LWRES_R_SUCCESS)
fatal("lwres_context_create failed");
if (lwresult != LWRES_R_SUCCESS)
"An error was encountered in /etc/resolv.conf\n");
if (ns_total <= 0)
fatal("no valid servers found");
fatal("out of memory");
for (i = 0; i < ns_total; i++) {
} else {
}
}
&dispatchv4);
NULL);
}
else
setup_key();
}
static void
ddebug("get_address()");
else {
fatal("Couldn't look up your server host %s. errno=%d",
port);
}
}
static void
int ch;
debug("parse_args");
switch (ch) {
case 'd':
break;
case 'D': /* was -dd */
break;
case 'M': /* was -dm */
break;
case 'y':
break;
case 'v':
break;
case 'k':
break;
default:
"[-y keyname:secret | -k keyfile] [-v]\n");
exit(1);
}
}
argv[0]);
exit(1);
}
}
static isc_uint16_t
char *word;
unsigned int dots;
dns_name_t *rn;
if (*word == 0) {
return (STATUS_SYNTAX);
}
rn = dns_rootname;
else
return (STATUS_MORE);
}
static isc_uint16_t
{
dns_name_t *rn;
cmdline++;
if (*cmdline != 0) {
else
&callbacks);
if (result != ISC_R_SUCCESS)
return (STATUS_MORE);
}
return (STATUS_MORE);
}
static isc_uint16_t
char *word;
ddebug("make_prereq()");
/*
* Read the owner name
*/
if (retval != STATUS_MORE)
return (retval);
/*
* If this is an rrset prereq, read the class or type.
*/
if (isrrset) {
if (*word == 0) {
return (STATUS_SYNTAX);
}
if (result == ISC_R_SUCCESS) {
/*
* Now read the type.
*/
if (*word == 0) {
return (STATUS_SYNTAX);
}
} else {
}
} else
if (isrrset && ispositive) {
if (retval != STATUS_MORE)
return (retval);
}
if (ispositive) {
else
} else
return (STATUS_MORE);
}
static isc_uint16_t
evaluate_prereq(char *cmdline) {
char *word;
ddebug("evaluate_prereq()");
if (*word == 0) {
return (STATUS_SYNTAX);
}
} else {
return (STATUS_SYNTAX);
}
}
static isc_uint16_t
evaluate_server(char *cmdline) {
if (*word == 0) {
return (STATUS_SYNTAX);
}
if (*word == 0)
else {
char *endp;
if (*endp != 0) {
return (STATUS_SYNTAX);
}
}
if (userserver == NULL) {
if (userserver == NULL)
fatal("out of memory");
}
return (STATUS_MORE);
}
static isc_uint16_t
evaluate_zone(char *cmdline) {
char *word;
isc_buffer_t b;
if (*word == 0) {
return (STATUS_SYNTAX);
}
NULL);
if (result != ISC_R_SUCCESS) {
return (STATUS_SYNTAX);
}
return (STATUS_MORE);
}
static isc_uint16_t
char *word;
char *endp;
ddebug("update_addordelete()");
/*
* Read the owner name
*/
if (retval != STATUS_MORE)
return (retval);
/*
* If this is an add, read the TTL and verify that it's numeric.
*/
if (!isdelete) {
if (*word == 0) {
goto failure;
}
if (*endp != 0) {
goto failure;
}
} else
ttl = 0;
/*
* Read the class or type.
*/
if (*word == 0) {
if (isdelete) {
goto doneparsing;
} else {
goto failure;
}
}
if (result == ISC_R_SUCCESS) {
/*
* Now read the type.
*/
if (*word == 0) {
if (isdelete) {
goto doneparsing;
} else {
goto failure;
}
}
} else {
}
&rdata);
if (retval != STATUS_MORE)
goto failure;
if (isdelete) {
else
} else {
goto failure;
}
}
return (STATUS_MORE);
return (STATUS_SYNTAX);
}
static isc_uint16_t
evaluate_update(char *cmdline) {
char *word;
ddebug("evaluate_update()");
if (*word == 0) {
return (STATUS_SYNTAX);
}
else {
return (STATUS_SYNTAX);
}
}
static void
show_message(void) {
ddebug("show_message()");
if (result != ISC_R_SUCCESS) {
return;
}
printf("Outgoing update query:\n%.*s",
(int)isc_buffer_usedlength(&buf),
(char*)isc_buffer_base(&buf));
}
static isc_uint16_t
get_next_command(void) {
char cmdlinebuf[MAXCMD];
char *cmdline;
char *word;
ddebug("get_next_command()");
return (STATUS_QUIT);
if (*word == 0)
return (STATUS_SEND);
return (STATUS_QUIT);
return (evaluate_prereq(cmdline));
return (evaluate_update(cmdline));
return (evaluate_server(cmdline));
return (evaluate_zone(cmdline));
return (STATUS_SEND);
show_message();
return (STATUS_MORE);
}
return (STATUS_SYNTAX);
}
static isc_boolean_t
user_interaction(void) {
ddebug("user_interaction()");
result = get_next_command();
if (result == STATUS_SEND)
return (ISC_TRUE);
return (ISC_FALSE);
}
static void
if (acquirelock)
if (acquirelock)
}
static void
ddebug("updated_completed()");
goto done;
}
if (debugging) {
(int)isc_buffer_usedlength(&buf),
(char*)isc_buffer_base(&buf));
}
done:
}
static void
unsigned int options = 0;
ddebug("send_update()");
if (usevc)
}
static void
int pass = 0;
ddebug("recvsoa()");
if (eresult != ISC_R_SUCCESS) {
char addrbuf[ISC_SOCKADDR_FORMATSIZE];
if (userserver != NULL)
fatal("Couldn't talk to specified nameserver.");
fatal("Couldn't talk to any default nameserver.");
return;
}
ddebug("About to create rcvmsg");
if (debugging) {
(int)isc_buffer_usedlength(&buf),
(char*)isc_buffer_base(&buf));
}
fatal("response to SOA query was unsuccessful");
if (pass == 0)
else if (pass == 1)
else
fatal("response to SOA query didn't contain an SOA");
if (result != ISC_R_SUCCESS) {
pass++;
goto lookforsoa;
}
while (result == ISC_R_SUCCESS) {
&soaset);
if (result == ISC_R_SUCCESS)
break;
}
pass++;
goto lookforsoa;
}
if (debugging) {
}
else
if (debugging) {
}
if (userserver != NULL)
else {
serveraddr = &tempaddr;
}
ddebug("Out of recvsoa");
}
static void
{
fatal("out of memory");
}
static void
start_update(void) {
ddebug("start_update()");
return;
}
if (result != ISC_R_SUCCESS) {
return;
}
&soaquery);
if (userserver != NULL)
else {
ns_inuse = 0;
}
}
static void
cleanup(void) {
ddebug("cleanup()");
if (userserver != NULL)
debug("Freeing key");
}
if (is_dst_up) {
debug("Destroy DST lib");
}
debug("Detach from entropy");
}
ddebug("Shutting down request manager");
ddebug("Freeing the dispatcher");
ddebug("Shutting down dispatch manager");
ddebug("Ending task");
ddebug("Shutting down task manager");
ddebug("Shutting down socket manager");
ddebug("Shutting down timer manager");
ddebug("Destroying memory context");
if (isc_mem_debugging)
}
int
setup_system();
while (ISC_TRUE) {
reset_system();
if (!user_interaction())
break;
start_update();
while (busy)
}
cleanup();
return (0);
}