nsupdate.c revision dbc41c829031c1ba7a4bade35b10474fab02b0a1
/*
* 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.63 2000/11/22 02:54:15 bwelling Exp $ */
#include <config.h>
#include <ctype.h>
#include <errno.h>
#include <limits.h>
#include <netdb.h>
#include <stdlib.h>
#include <unistd.h>
#if (!(defined(HAVE_ADDRINFO) && defined(HAVE_GETADDRINFO)))
extern int h_errno;
#endif
#include <isc/commandline.h>
#include <isc/sockaddr.h>
#include <dns/callbacks.h>
#include <dns/dispatch.h>
#include <dns/fixedname.h>
#include <dns/rdataclass.h>
#include <dns/rdatalist.h>
#include <dns/rdataset.h>
#include <dns/rdatastruct.h>
#include <dns/rdatatype.h>
#define NAMEBUF 512
#define WORDLEN 512
#define FIND_TIMEOUT 5
#define DNSDEFAULTPORT 53
#define RESOLV_CONF "/etc/resolv.conf"
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 {
static void
#define STATUS_MORE (isc_uint16_t)0
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(void) {
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
ddebug("shutdown_program()");
}
static void
setup_system(void) {
int i;
ddebug("setup_system()");
result = isc_net_probeipv4();
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 %s\n", RESOLV_CONF);
if (ns_total <= 0)
fatal("no valid servers found");
fatal("out of memory");
for (i = 0; i < ns_total; i++) {
} else {
}
}
if (have_ipv6) {
4, 2, 3, 5,
}
NULL);
}
else
setup_key();
}
static void
#if defined(HAVE_ADDRINFO) && defined(HAVE_GETADDRINFO)
int result;
#else
#endif
ddebug("get_address()");
else {
#if defined(HAVE_ADDRINFO) && defined(HAVE_GETADDRINFO)
if (result != 0) {
fatal("Couldn't find server '%s': %s",
}
#else
fatal("Couldn't look up your server host %s. errno=%d",
port);
#endif
}
}
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
do {
"space for the rdata\n");
exit(1);
}
bufsz *= 2;
} while (result == ISC_R_NOSPACE);
if (result != ISC_R_SUCCESS)
return (STATUS_MORE);
} else {
}
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) {
long port;
if (*word == 0) {
return (STATUS_SYNTAX);
}
if (*word == 0)
else {
char *endp;
if (*endp != 0) {
return (STATUS_SYNTAX);
"(1 to 65535)\n", word);
return (STATUS_SYNTAX);
}
}
if (userserver == NULL) {
if (userserver == NULL)
fatal("out of memory");
}
return (STATUS_MORE);
}
static isc_uint16_t
evaluate_local(char *cmdline) {
long port;
if (*word == 0) {
return (STATUS_SYNTAX);
}
if (*word == 0)
port = 0;
else {
char *endp;
if (*endp != 0) {
return (STATUS_SYNTAX);
"(1 to 65535)\n", word);
return (STATUS_SYNTAX);
}
}
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
long ttl;
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 in range.
*/
if (!isdelete) {
if (*word == 0) {
goto failure;
}
if (*endp != '\0') {
goto failure;
/*
* The errno test is needed to catch when strtol()
* overflows on a platform where sizeof(int) ==
* sizeof(long), because ttl will be set to LONG_MAX,
* which will be equal to TTL_MAX.
*/
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
int bufsz;
ddebug("show_message()");
do {
"buffer to display message\n");
exit(1);
}
bufsz *= 2;
} while (result == ISC_R_NOSPACE);
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);
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_local(cmdline));
return (evaluate_zone(cmdline));
return (STATUS_SEND);
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
done_update(void) {
ddebug("done_update()");
}
static void
ddebug("update_completed()");
goto done;
}
if (debugging) {
int bufsz;
do {
"enough buffer to display message\n");
exit(1);
}
bufsz *= 2;
} while (result == ISC_R_NOSPACE);
(int)isc_buffer_usedlength(buf),
(char*)isc_buffer_base(buf));
}
done:
done_update();
}
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 bufsz;
do {
"space for debugging message\n");
exit(1);
}
} while (result == ISC_R_NOSPACE);
(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) {
char namestr[DNS_NAME_FORMATSIZE];
}
else
if (debugging) {
char namestr[DNS_NAME_FORMATSIZE];
}
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) {
done_update();
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 dispatchers");
if (have_ipv6)
ddebug("Shutting down dispatch manager");
ddebug("Shutting down task manager");
ddebug("Destroying event");
ddebug("Shutting down socket manager");
ddebug("Shutting down timer manager");
ddebug("Destroying memory context");
if (memdebugging)
}
static void
if (global_event == NULL)
if (shuttingdown) {
return;
}
reset_system();
more = user_interaction();
if (!more) {
return;
}
start_update();
return;
}
int
setup_system();
(void)isc_app_run();
cleanup();
return (0);
}