📄 test_sresolv.c
字号:
sres_query_make(res, test_answer_multi, ctx, s, sres_qtype_any, "example.com"); sres_query_make(res, test_answer_multi, ctx, s, sres_qtype_any, "_sips._tcp.example.com"); sres_query_make(res, test_answer_multi, ctx, s, sres_qtype_any, "sip.example.com"); sres_query_make(res, test_answer_multi, ctx, s, sres_qtype_any, "subnet.example.com");#if HAVE_SIN6 sres_query_make(res, test_answer_multi, ctx, s, sres_type_aaaa, "mgw02.example.com"); inet_pton(AF_INET6, "3ffe:1200:3012:c000:0a08:20ff:fe7d:e7ac", &sin6.sin6_addr); sin6.sin6_family = AF_INET6; query = sres_query_make_sockaddr(res, test_answer_multi, ctx, s, sres_qtype_any, (struct sockaddr *)&sin6); TEST_1(query != NULL);#endif TEST_RUN(ctx); /* For a chance, a fully qualified domain name with final "." */ domain = "example.com."; result = sres_cached_answers(res, sres_qtype_any, domain); TEST_1(result != NULL); for (i = 0; result[i] != NULL; i++) { rr = result[i]; if (rr->sr_record->r_type == sres_type_naptr) { sres_naptr_record_t *rr_naptr = rr->sr_naptr; switch(rr_naptr->na_order) { case 20: TEST(rr_naptr->na_record->r_type, sres_type_naptr); TEST(rr_naptr->na_record->r_class, sres_class_in); TEST(rr_naptr->na_record->r_ttl, 60); TEST(rr_naptr->na_order, 20); TEST(rr_naptr->na_prefer, 50); TEST_S(rr_naptr->na_flags, "s"); TEST_S(rr_naptr->na_services, "SIPS+D2T"); TEST_S(rr_naptr->na_regexp, ""); TEST_S(rr_naptr->na_replace, "_sips._tcp.example.com."); ok |= 1; break; case 40: TEST(rr_naptr->na_record->r_type, sres_type_naptr); TEST(rr_naptr->na_record->r_class, sres_class_in); TEST(rr_naptr->na_record->r_ttl, 60); TEST(rr_naptr->na_order, 40); TEST(rr_naptr->na_prefer, 15); TEST_S(rr_naptr->na_flags, "s"); TEST_S(rr_naptr->na_services, "SIP+D2U"); TEST_S(rr_naptr->na_regexp, ""); TEST_S(rr_naptr->na_replace, "_sip._udp.example.com."); ok |= 2; break; case 50: TEST(rr_naptr->na_record->r_type, sres_type_naptr); TEST(rr_naptr->na_record->r_class, sres_class_in); // TEST(rr_naptr->na_record->r_ttl, 60); TEST(rr_naptr->na_order, 50); TEST(rr_naptr->na_prefer, 15); TEST_S(rr_naptr->na_flags, "u"); TEST_S(rr_naptr->na_services, "TEST+D2U"); TEST_S(rr_naptr->na_regexp, "/(tst:([^@]+@))?example.com$/\\1operator.com/i"); TEST_S(rr_naptr->na_replace, "."); break; case 80: TEST(rr_naptr->na_record->r_type, sres_type_naptr); TEST(rr_naptr->na_record->r_class, sres_class_in); TEST(rr_naptr->na_record->r_ttl, 60); TEST(rr_naptr->na_order, 80); TEST(rr_naptr->na_prefer, 25); TEST_S(rr_naptr->na_flags, "s"); TEST_S(rr_naptr->na_services, "SIP+D2T"); TEST_S(rr_naptr->na_regexp, ""); TEST_S(rr_naptr->na_replace, "_sip._tcp.example.com."); ok |= 4; break; default: TEST_1(0); } } } TEST(ok, 7); /* Reverse order before sorting */ for (j = 0; j < --i; j++) { sres_record_t *swap = result[j]; result[j] = result[i]; result[i] = swap; } /* Test sorting */ sres_sort_answers(res, result); /* Sort all records with themselves */ for (i = 0; result[i]; i++) { sort_array[0] = result[i], sort_array[1] = result[i], sort_array[2] = NULL; sres_sort_answers(res, sort_array); } /* Test free */ for (i = 0; result[i]; i++) { sres_free_answer(res, result[i]); result[i] = NULL; } /* Test sres_free_answers() */ sres_free_answers(res, result); result = sres_cached_answers(res, sres_qtype_any, "_sips._tcp.example.com"); TEST_1(result != NULL); ok = 0; for (i = 0; result[i] != NULL; i++) { sres_srv_record_t *rr_srv = result[i]->sr_srv; TEST(rr_srv->srv_record->r_type, sres_type_srv); switch(rr_srv->srv_priority) { case 3: TEST(rr_srv->srv_record->r_type, sres_type_srv); TEST(rr_srv->srv_record->r_class, sres_class_in); TEST(rr_srv->srv_record->r_ttl, 60); TEST(rr_srv->srv_weight, 100); TEST(rr_srv->srv_port, 5061); TEST_S(rr_srv->srv_target, "sip00.example.com."); ok |= 1; break; case 4: TEST(rr_srv->srv_record->r_type, sres_type_srv); TEST(rr_srv->srv_record->r_class, sres_class_in); TEST(rr_srv->srv_record->r_ttl, 60); TEST(rr_srv->srv_weight, 50); TEST(rr_srv->srv_port, 5050); TEST_S(rr_srv->srv_target, "sip02.example.com."); ok |= 2; break; case 5: TEST(rr_srv->srv_record->r_type, sres_type_srv); TEST(rr_srv->srv_record->r_class, sres_class_in); TEST(rr_srv->srv_record->r_ttl, 60); TEST(rr_srv->srv_weight, 10); TEST(rr_srv->srv_port, 5060); TEST_S(rr_srv->srv_target, "sip01.example.com."); ok |= 4; break; default: TEST_1(0); } } TEST(ok, 7); /* Reverse order before sorting */ for (j = 0; j < --i; j++) { sres_record_t *swap = result[j]; result[j] = result[i]; result[i] = swap; } sres_sort_answers(res, result); sres_free_answers(res, result); domain = "sip.example.com"; result = sres_cached_answers(res, sres_qtype_any, domain); TEST_1(result != NULL); TEST_1(result[0] != NULL); rr_cname = result[0]->sr_cname; TEST(rr_cname->cname_record->r_type, sres_type_cname); TEST(rr_cname->cname_record->r_class, sres_class_in); TEST(rr_cname->cname_record->r_ttl, 60); TEST_S(rr_cname->cn_cname, "sip00.example.com."); // XXX - this also returns AAAA record for sip00 //TEST(result[1], NULL); sres_free_answers(res, result);#if HAVE_SIN6 domain = "subnet.example.com"; result = sres_cached_answers(res, sres_qtype_any, domain); TEST_1(result != NULL); TEST_1(result[0] != NULL); rr_a6 = result[0]->sr_a6; TEST(rr_a6->a6_record->r_type, sres_type_a6); TEST(rr_a6->a6_record->r_class, sres_class_in); TEST(rr_a6->a6_record->r_ttl, 60); TEST(rr_a6->a6_prelen, 0); TEST_S(inet_ntop(AF_INET6, &rr_a6->a6_suffix, buf, sizeof(buf)), "3ff0::"); TEST(rr_a6->a6_prename, NULL); sres_free_answers(res, result); domain = "mgw02.example.com"; TEST_1(result = sres_cached_answers(res, sres_type_aaaa, domain)); TEST_1(rr_aaaa = result[0]->sr_aaaa); TEST(rr_aaaa->aaaa_record->r_type, sres_type_aaaa); TEST(rr_aaaa->aaaa_record->r_class, sres_class_in); TEST(rr_aaaa->aaaa_record->r_ttl, 60); TEST_S(inet_ntop(AF_INET6, &rr_aaaa->aaaa_addr, buf, sizeof(buf)), "3ffe:1200:3012:c000:206:5bff:fe55:462f"); sres_free_answers(res, result); result = sres_cached_answers_sockaddr(res, sres_type_ptr, (struct sockaddr *)&sin6); TEST_1(result != NULL); rr_ptr = result[0]->sr_ptr; TEST_1(rr_ptr != NULL); TEST(rr_ptr->ptr_record->r_type, sres_type_ptr); TEST(rr_ptr->ptr_record->r_class, sres_class_in); TEST_S(rr_ptr->ptr_domain, "sip01.example.com."); sres_free_answers(res, result);#endif /* HAVE_SIN6 */ END();}#if HAVE_SIN6int test_query_one_type(sres_context_t *ctx){ sres_resolver_t *res = ctx->resolver; int s = ctx->socket; sres_record_t **result; const sres_aaaa_record_t *rr_aaaa; char buf[50] = {0}; BEGIN(); TEST_1(sres_query_make(res, test_answer, ctx, s, sres_type_aaaa, "mgw02.example.com")); TEST_RUN(ctx); TEST_1(result = ctx->result); TEST_1(result[0] != NULL); TEST_1(rr_aaaa = result[0]->sr_aaaa); TEST(rr_aaaa->aaaa_record->r_type, sres_type_aaaa); TEST(rr_aaaa->aaaa_record->r_class, sres_class_in); TEST(rr_aaaa->aaaa_record->r_ttl, 60); TEST_S(inet_ntop(AF_INET6, &rr_aaaa->aaaa_addr, buf, sizeof(buf)), "3ffe:1200:3012:c000:206:5bff:fe55:462f"); TEST(result[1], NULL); END();}#endif /* HAVE_SIN6*/#ifdef _WIN32#include <fcntl.h>#endifint sink_make(sres_context_t *ctx){ char *tmpdir = getenv("TMPDIR"); char *template; int fd, sink; struct sockaddr_in sin[1]; socklen_t sinsize = sizeof sin; FILE *f; BEGIN(); sink = socket(AF_INET, SOCK_DGRAM, 0); TEST_1(sink != -1); TEST(getsockname(sink, (struct sockaddr *)sin, &sinsize), 0); sin->sin_addr.s_addr = htonl(INADDR_LOOPBACK); TEST(bind(sink, (struct sockaddr *)sin, sinsize), 0); TEST(getsockname(sink, (struct sockaddr *)sin, &sinsize), 0); ctx->sink = sink; template = su_sprintf(ctx->home, "%s/test_sresolv.XXXXXX", tmpdir ? tmpdir : "/tmp"); TEST_1(template);#ifndef _WIN32 fd = mkstemp(template); TEST_1(fd != -1);#else fd = open(template, O_WRONLY); TEST_1(fd != -1);#endif f = fdopen(fd, "w"); TEST_1(f); fprintf(f, "domain example.com\n" "search foo.bar.com\n" "port %u\n", ntohs(sin->sin_port)); fclose(f); ctx->sinkconf = template; END();}int recv_sink(su_root_magic_t *rm, su_wait_t *w, sres_context_t *ctx){ union { char bytes[8192]; unsigned short shorts[4096]; } buffer[1]; su_wait_events(w, ctx->sink); recv(ctx->sink, buffer->bytes, sizeof buffer, 0); return 0;}int sink_init(sres_context_t *ctx){ su_wait_t w[1]; BEGIN(); TEST(su_wait_create(w, ctx->sink, SU_WAIT_IN), 0); ctx->sinkidx = su_root_register(ctx->root, w, recv_sink, ctx, 0); TEST_1(ctx->sinkidx != 0); END();}int sink_deinit(sres_context_t *ctx){ BEGIN(); if (ctx->sinkidx) su_root_deregister(ctx->root, ctx->sinkidx); ctx->sinkidx = 0; su_close(ctx->sink), ctx->sink = -1; END();}int test_timeout(sres_context_t *ctx){ sres_resolver_t *res = ctx->resolver; int s = ctx->socket; sres_record_t **result; /* const sres_soa_record_t *rr_soa; */ char const *domain = "test"; BEGIN(); sres_query_make(res, test_answer, ctx, s, sres_type_a, domain); ctx->timeout = 1; TEST_RUN(ctx); ctx->timeout = 0; TEST(ctx->result, NULL); result = sres_cached_answers(res, sres_type_a, domain);#if 0 TEST_1(result); TEST_1(result[0] != NULL); rr_soa = result[0]->sr_soa; TEST(rr_soa->soa_record->r_type, sres_type_soa); TEST(rr_soa->soa_record->r_class, sres_class_in); TEST_S(rr_soa->soa_record->r_name, "example.com."); TEST(rr_soa->soa_record->r_status, SRES_TIMEOUT_ERR); sres_free_answers(res, result);#else /* Currently, we do not create error records */ TEST_1(result == NULL);#endif END();}void test_answer(sres_context_t *ctx, sres_query_t *q, sres_record_t **answer){ ctx->query = q; if (ctx->result) sres_free_answers(ctx->resolver, ctx->result); ctx->result = answer; BREAK(ctx);}void test_answer_multi(sres_context_t *ctx, sres_query_t *q, sres_record_t **answer){ static int count = 0; ctx->query = q; count++; sres_free_answers(ctx->resolver, answer); if (count == 6) BREAK(ctx);}#include <sys/time.h>/* Fake time() implementation */time_t time(time_t *tp){ struct timeval tv[1];#ifndef _WIN32 gettimeofday(tv, NULL);#else return 0;#endif if (tp) *tp = tv->tv_sec + offset; return tv->tv_sec + offset;}int test_expiration(sres_context_t *ctx){ sres_resolver_t *res = ctx->resolver; sres_record_t **result; char const *domain = "example.com"; BEGIN(); offset = 3600; /* Time suddenly proceeds by an hour.. */ sres_resolver_timer(res, ctx->socket); result = sres_cached_answers(res, sres_qtype_any, domain); TEST(result, NULL); /* the cache should be empty after 15 secs */ END();}#define is_hexdigit(c) ((c >= '0' && c <= '9') || (c >= 'a' && c <= 'f'))#define hex(c) ((c >= '0' && c <= '9') ? (c - '0') : (c - 'a' + 10))/* Convert lowercase hex to binary */staticvoid *hex2bin(char const *test_name, char const *hex1, char const *hex2, unsigned *binsize){ char output[2048]; char *bin; char const *b; int j; if (hex1 == NULL || binsize == NULL) return NULL; for (b = hex1, j = 0; b;) { while (b[0]) { if (is_hexdigit(b[0])) { if (!is_hexdigit(b[1])) { fprintf(stderr, "%s: hex2bin: invalid hex '%c'\n", test_name, b[1]); exit(2); } output[j++] = (hex(b[0]) << 4) | hex(b[1]); if (j == sizeof(output)) { fprintf(stderr, "%s:%u: hex2bin: buffer too small\n",
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -