📄 t_rbt.c
字号:
/* * Copyright (C) 2004 Internet Systems Consortium, Inc. ("ISC") * Copyright (C) 1998-2001, 2003 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 ISC DISCLAIMS ALL WARRANTIES WITH * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY * AND FITNESS. IN NO EVENT SHALL ISC 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: t_rbt.c,v 1.23.206.4 2004/10/25 01:36:07 marka Exp $ */#include <config.h>#include <ctype.h>#include <stdlib.h>#include <isc/entropy.h>#include <isc/mem.h>#include <isc/util.h>#include <isc/hash.h>#include <isc/string.h>#include <dns/fixedname.h>#include <dns/rbt.h>#include <dns/result.h>#include <tests/t_api.h>#define BUFLEN 1024#define DNSNAMELEN 255char *progname;char *Tokens[T_MAXTOKS];static intt_dns_rbtnodechain_init(char *dbfile, char *findname, char *firstname, char *firstorigin, char *nextname, char *nextorigin, char *prevname, char *prevorigin, char *lastname, char *lastorigin);static char *fixedname_totext(dns_fixedname_t *name);static intfixedname_cmp(dns_fixedname_t *dns_name, char *txtname);static char *dnsname_totext(dns_name_t *name);static intt_namechk(isc_result_t dns_result, dns_fixedname_t *dns_name, char *exp_name, dns_fixedname_t *dns_origin, char *exp_origin, isc_result_t exp_result);/* * Parts adapted from the original rbt_test.c. */static intfixedname_cmp(dns_fixedname_t *dns_name, char *txtname) { char *name; name = dnsname_totext(dns_fixedname_name(dns_name)); if (strcmp(txtname, "NULL") == 0) { if ((name == NULL) || (*name == '\0')) return(0); return(1); } else { return(strcmp(name, txtname)); }}static char *dnsname_totext(dns_name_t *name) { static char buf[BUFLEN]; isc_buffer_t target; isc_buffer_init(&target, buf, BUFLEN); dns_name_totext(name, ISC_FALSE, &target); *((char *)(target.base) + target.used) = '\0'; return(target.base);}static char *fixedname_totext(dns_fixedname_t *name) { static char buf[BUFLEN]; isc_buffer_t target; memset(buf, 0, BUFLEN); isc_buffer_init(&target, buf, BUFLEN); dns_name_totext(dns_fixedname_name(name), ISC_FALSE, &target); *((char *)(target.base) + target.used) = '\0'; return(target.base);}#ifdef NEED_PRINT_DATAstatic isc_result_tprint_data(void *data) { isc_result_t dns_result; isc_buffer_t target; char *buffer[DNSNAMELEN]; isc_buffer_init(&target, buffer, sizeof(buffer)); dns_result = dns_name_totext(data, ISC_FALSE, &target); if (dns_result != ISC_R_SUCCESS) { t_info("dns_name_totext failed %s\n", dns_result_totext(dns_result)); } return(dns_result);}#endif /* NEED_PRINT_DATA */static intcreate_name(char *s, isc_mem_t *mctx, dns_name_t **dns_name) { int nfails; int length; isc_result_t result; isc_buffer_t source; isc_buffer_t target; nfails = 0; if (s && *s) { length = strlen(s); isc_buffer_init(&source, s, length); isc_buffer_add(&source, length); /* * The buffer for the actual name will immediately follow the * name structure. */ *dns_name = isc_mem_get(mctx, sizeof(**dns_name) + DNSNAMELEN); if (*dns_name == NULL) { t_info("isc_mem_get failed\n"); ++nfails; } dns_name_init(*dns_name, NULL); isc_buffer_init(&target, *dns_name + 1, DNSNAMELEN); result = dns_name_fromtext(*dns_name, &source, dns_rootname, ISC_FALSE, &target); if (result != ISC_R_SUCCESS) { ++nfails; t_info("dns_name_fromtext(%s) failed %s\n", s, dns_result_totext(result)); } } else { ++nfails; t_info("create_name: empty name\n"); } return(nfails);}static voiddelete_name(void *data, void *arg) { isc_mem_put((isc_mem_t *)arg, data, sizeof(dns_name_t) + DNSNAMELEN);}/* * Adapted from the original rbt_test.c. */static intt1_add(char *name, dns_rbt_t *rbt, isc_mem_t *mctx, isc_result_t *dns_result) { int nprobs; dns_name_t *dns_name; nprobs = 0; if (name && dns_result) { *dns_result = create_name(name, mctx, &dns_name); if (*dns_result == ISC_R_SUCCESS) { if (T_debug) t_info("dns_rbt_addname succeeded\n"); *dns_result = dns_rbt_addname(rbt, dns_name, dns_name); } else { t_info("dns_rbt_addname failed %s\n", dns_result_totext(*dns_result)); delete_name(dns_name, mctx); ++nprobs; } } else { ++nprobs; } return(nprobs);}static intt1_delete(char *name, dns_rbt_t *rbt, isc_mem_t *mctx, isc_result_t *dns_result){ int nprobs; dns_name_t *dns_name; nprobs = 0; if (name && dns_result) { *dns_result = create_name(name, mctx, &dns_name); if (*dns_result == ISC_R_SUCCESS) { *dns_result = dns_rbt_deletename(rbt, dns_name, ISC_FALSE); delete_name(dns_name, mctx); } else { ++nprobs; } } else { ++nprobs; } return(nprobs);}static intt1_search(char *name, dns_rbt_t *rbt, isc_mem_t *mctx, isc_result_t *dns_result){ int nprobs; dns_name_t *dns_searchname; dns_name_t *dns_foundname; dns_fixedname_t dns_fixedname; void *data; nprobs = 0; if (name && dns_result) { *dns_result = create_name(name, mctx, &dns_searchname); if (*dns_result == ISC_R_SUCCESS) { dns_fixedname_init(&dns_fixedname); dns_foundname = dns_fixedname_name(&dns_fixedname); data = NULL; *dns_result = dns_rbt_findname(rbt, dns_searchname, 0, dns_foundname, &data); delete_name(dns_searchname, mctx); } else { ++nprobs; } } else { ++nprobs; } return(nprobs);}/* * Initialize a database from filename. */static intrbt_init(char *filename, dns_rbt_t **rbt, isc_mem_t *mctx) { int rval; isc_result_t dns_result; char *p; FILE *fp; fp = fopen(filename, "r"); if (fp == NULL) { t_info("No such file %s\n", filename); return(1); } dns_result = dns_rbt_create(mctx, delete_name, mctx, rbt); if (dns_result != ISC_R_SUCCESS) { t_info("dns_rbt_create failed %s\n", dns_result_totext(dns_result)); fclose(fp); return(1); } while ((p = t_fgetbs(fp)) != NULL) { /* * Skip any comment lines. */ if ((*p == '#') || (*p == '\0') || (*p == ' ')) { free(p); continue; } if (T_debug) t_info("adding name %s to the rbt\n", p); rval = t1_add(p, *rbt, mctx, &dns_result); if ((rval != 0) || (dns_result != ISC_R_SUCCESS)) { t_info("add of %s failed\n", p); dns_rbt_destroy(rbt); fclose(fp); return(1); } (void) free(p); } fclose(fp); return(0);}static inttest_rbt_gen(char *filename, char *command, char *testname, isc_result_t exp_result){ int rval; int result; dns_rbt_t *rbt; isc_result_t isc_result; isc_result_t dns_result; isc_mem_t *mctx; isc_entropy_t *ectx; dns_name_t *dns_name; result = T_UNRESOLVED; if (strcmp(command, "create") != 0) t_info("testing using name %s\n", testname); mctx = NULL; ectx = NULL; isc_result = isc_mem_create(0, 0, &mctx); if (isc_result != ISC_R_SUCCESS) { t_info("isc_mem_create: %s: exiting\n", dns_result_totext(isc_result)); return(T_UNRESOLVED); } isc_result = isc_entropy_create(mctx, &ectx); if (isc_result != ISC_R_SUCCESS) { t_info("isc_entropy_create: %s: exiting\n", dns_result_totext(isc_result)); isc_mem_destroy(&mctx); return(T_UNRESOLVED); } isc_result = isc_hash_create(mctx, ectx, DNS_NAME_MAXWIRE); if (isc_result != ISC_R_SUCCESS) { t_info("isc_hash_create: %s: exiting\n", dns_result_totext(isc_result)); isc_entropy_detach(&ectx); isc_mem_destroy(&mctx); return(T_UNRESOLVED); } rbt = NULL; if (rbt_init(filename, &rbt, mctx) != 0) { if (strcmp(command, "create") == 0) result = T_FAIL; isc_hash_destroy(); isc_entropy_detach(&ectx); isc_mem_destroy(&mctx); return(result); } /* * Now try the database command. */ if (strcmp(command, "create") == 0) { result = T_PASS; } else if (strcmp(command, "add") == 0) { dns_result = create_name(testname, mctx, &dns_name); if (dns_result == ISC_R_SUCCESS) { dns_result = dns_rbt_addname(rbt, dns_name, dns_name); if (dns_result != ISC_R_SUCCESS) delete_name(dns_name, mctx); if (dns_result == exp_result) { if (dns_result == ISC_R_SUCCESS) { rval = t1_search(testname, rbt, mctx, &dns_result); if (rval == 0) { if (dns_result == ISC_R_SUCCESS) { result = T_PASS; } else { result = T_FAIL; } } else { t_info("t1_search failed\n"); result = T_UNRESOLVED; } } else { result = T_PASS; } } else { t_info("dns_rbt_addname returned %s, " "expected %s\n", dns_result_totext(dns_result), dns_result_totext(exp_result)); result = T_FAIL; } } else { t_info("create_name failed %s\n", dns_result_totext(dns_result)); result = T_UNRESOLVED; } } else if ((strcmp(command, "delete") == 0) || (strcmp(command, "nuke") == 0)) { rval = t1_delete(testname, rbt, mctx, &dns_result); if (rval == 0) { if (dns_result == exp_result) { rval = t1_search(testname, rbt, mctx, &dns_result); if (rval == 0) { if (dns_result == ISC_R_SUCCESS) { t_info("dns_rbt_deletename " "didn't delete " "the name"); result = T_FAIL; } else { result = T_PASS; } } } else { t_info("delete returned %s, expected %s\n", dns_result_totext(dns_result), dns_result_totext(exp_result)); result = T_FAIL; } } } else if (strcmp(command, "search") == 0) { rval = t1_search(testname, rbt, mctx, &dns_result); if (rval == 0) { if (dns_result == exp_result) { result = T_PASS; } else { t_info("find returned %s, expected %s\n", dns_result_totext(dns_result), dns_result_totext(exp_result)); result = T_FAIL; } } } dns_rbt_destroy(&rbt); isc_hash_destroy(); isc_entropy_detach(&ectx); isc_mem_destroy(&mctx); return(result);}static inttest_dns_rbt_x(const char *filename) { FILE *fp; char *p; int line; int cnt; int result; int nfails; int nprobs; nfails = 0; nprobs = 0; fp = fopen(filename, "r"); if (fp != NULL) { line = 0; while ((p = t_fgetbs(fp)) != NULL) { ++line; /* * Skip comment lines. */ if ((isspace((unsigned char)*p)) || (*p == '#')) continue; /* * Name of db file, command, testname, * expected result. */ cnt = t_bustline(p, Tokens); if (cnt == 4) { result = test_rbt_gen(Tokens[0], Tokens[1], Tokens[2], t_dns_result_fromtext(Tokens[3])); if (result != T_PASS) ++nfails; } else { t_info("bad format in %s at line %d\n", filename, line); ++nprobs; } (void)free(p); } (void)fclose(fp); } else { t_info("Missing datafile %s\n", filename); ++nprobs; } result = T_UNRESOLVED; if ((nfails == 0) && (nprobs == 0)) result = T_PASS; else if (nfails) result = T_FAIL; return(result);}static const char *a1 = "dns_rbt_create creates a rbt and returns " "ISC_R_SUCCESS on success";static voidt1() { int result; t_assert("dns_rbt_create", 1, T_REQUIRED, a1); result = test_dns_rbt_x("dns_rbt_create_1_data"); t_result(result);}static const char *a2 = "dns_rbt_addname adds a name to a database and " "returns ISC_R_SUCCESS on success";static voidt2() { int result; t_assert("dns_rbt_addname", 2, T_REQUIRED, a2); result = test_dns_rbt_x("dns_rbt_addname_1_data"); t_result(result);}static const char *a3 = "when name already exists, dns_rbt_addname() " "returns ISC_R_EXISTS";static voidt3() { int result; t_assert("dns_rbt_addname", 3, T_REQUIRED, a3); result = test_dns_rbt_x("dns_rbt_addname_2_data"); t_result(result);}static const char *a4 = "when name exists, dns_rbt_deletename() returns " "ISC_R_SUCCESS";static voidt4() { int result; t_assert("dns_rbt_deletename", 4, T_REQUIRED, a4); result = test_dns_rbt_x("dns_rbt_deletename_1_data"); t_result(result);}static const char *a5 = "when name does not exist, dns_rbt_deletename() " "returns ISC_R_NOTFOUND";static voidt5() { int result; t_assert("dns_rbt_deletename", 5, T_REQUIRED, a5); result = test_dns_rbt_x("dns_rbt_deletename_2_data"); t_result(result);}static const char *a6 = "when name exists and exactly matches the " "search name dns_rbt_findname() returns ISC_R_SUCCESS";static voidt6() { int result; t_assert("dns_rbt_findname", 6, T_REQUIRED, a6); result = test_dns_rbt_x("dns_rbt_findname_1_data"); t_result(result);}static const char *a7 = "when a name does not exist, " "dns_rbt_findname returns ISC_R_NOTFOUND";static voidt7() { int result; t_assert("dns_rbt_findname", 7, T_REQUIRED, a7); result = test_dns_rbt_x("dns_rbt_findname_2_data"); t_result(result);}static const char *a8 = "when a superdomain is found with data matching name, " "dns_rbt_findname returns DNS_R_PARTIALMATCH";static voidt8() { int result; t_assert("dns_rbt_findname", 8, T_REQUIRED, a8); result = test_dns_rbt_x("dns_rbt_findname_3_data"); t_result(result);}static const char *a9 = "a call to dns_rbtnodechain_init(chain, mctx) " "initializes chain";static intt9_walkchain(dns_rbtnodechain_t *chain, dns_rbt_t *rbt) { int cnt; int order; unsigned int nlabels; int nprobs; isc_result_t dns_result; dns_fixedname_t name;
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -