2014-08-07 23:14:10 +04:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2006-2014 Erik Ekman <yarrick@kryo.se>,
|
|
|
|
* 2006-2009 Bjorn Andersson <flex@kryo.se>
|
|
|
|
*
|
|
|
|
* Permission to use, copy, modify, and/or 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 THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
|
|
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
|
|
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR 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.
|
|
|
|
*/
|
|
|
|
|
2014-02-08 02:08:12 +04:00
|
|
|
#include <check.h>
|
|
|
|
#include <common.h>
|
2014-02-11 01:34:24 +04:00
|
|
|
#include <unistd.h>
|
2014-02-08 02:08:12 +04:00
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <netdb.h>
|
|
|
|
|
2014-05-30 02:18:45 +04:00
|
|
|
START_TEST(test_topdomain_ok)
|
|
|
|
{
|
2014-05-31 12:06:44 +04:00
|
|
|
char *error;
|
|
|
|
|
|
|
|
fail_if(check_topdomain("foo.0123456789.qwertyuiop.asdfghjkl.zxcvbnm.com", &error));
|
2014-05-30 02:18:45 +04:00
|
|
|
|
|
|
|
/* Not allowed to start with dot */
|
2014-05-31 12:06:44 +04:00
|
|
|
fail_unless(check_topdomain(".foo.0123456789.qwertyuiop.asdfghjkl.zxcvbnm.com", &error));
|
|
|
|
fail_if(strcmp("Starts with a dot", error));
|
|
|
|
|
|
|
|
/* Test missing error msg ptr */
|
|
|
|
fail_unless(check_topdomain(".foo", NULL));
|
2014-05-30 02:18:45 +04:00
|
|
|
}
|
|
|
|
END_TEST
|
|
|
|
|
|
|
|
START_TEST(test_topdomain_length)
|
|
|
|
{
|
2014-05-31 12:06:44 +04:00
|
|
|
char *error;
|
|
|
|
|
2014-05-30 02:18:45 +04:00
|
|
|
/* Test empty and too short */
|
2014-05-31 12:06:44 +04:00
|
|
|
fail_unless(check_topdomain("", &error));
|
|
|
|
fail_if(strcmp("Too short (< 3)", error));
|
|
|
|
fail_unless(check_topdomain("a", &error));
|
|
|
|
fail_if(strcmp("Too short (< 3)", error));
|
|
|
|
fail_unless(check_topdomain(".a", &error));
|
|
|
|
fail_if(strcmp("Too short (< 3)", error));
|
|
|
|
fail_unless(check_topdomain("a.", &error));
|
|
|
|
fail_if(strcmp("Too short (< 3)", error));
|
|
|
|
fail_unless(check_topdomain("ab", &error));
|
|
|
|
fail_if(strcmp("Too short (< 3)", error));
|
|
|
|
fail_if(check_topdomain("a.b", &error));
|
|
|
|
fail_if(strcmp("Too short (< 3)", error));
|
2014-05-30 02:18:45 +04:00
|
|
|
|
|
|
|
/* Test too long (over 128, need rest of space for data) */
|
|
|
|
fail_unless(check_topdomain(
|
|
|
|
"abcd12345.abcd12345.abcd12345.abcd12345.abcd12345."
|
|
|
|
"abcd12345.abcd12345.abcd12345.abcd12345.abcd12345."
|
2014-05-31 12:06:44 +04:00
|
|
|
"abcd12345.abcd12345.foo129xxx", &error));
|
|
|
|
fail_if(strcmp("Too long (> 128)", error));
|
2014-05-30 02:18:45 +04:00
|
|
|
fail_if(check_topdomain(
|
|
|
|
"abcd12345.abcd12345.abcd12345.abcd12345.abcd12345."
|
|
|
|
"abcd12345.abcd12345.abcd12345.abcd12345.abcd12345."
|
2014-05-31 12:06:44 +04:00
|
|
|
"abcd12345.abcd12345.foo128xx", &error));
|
2014-05-30 02:18:45 +04:00
|
|
|
}
|
|
|
|
END_TEST
|
|
|
|
|
|
|
|
START_TEST(test_topdomain_chunks)
|
|
|
|
{
|
2014-05-31 12:06:44 +04:00
|
|
|
char *error;
|
|
|
|
|
2014-05-30 02:18:45 +04:00
|
|
|
/* Must have at least one dot */
|
2014-05-31 12:06:44 +04:00
|
|
|
fail_if(check_topdomain("abcde.gh", &error));
|
|
|
|
fail_unless(check_topdomain("abcdefgh", &error));
|
|
|
|
fail_if(strcmp("No dots", error));
|
2014-05-30 02:18:45 +04:00
|
|
|
|
|
|
|
/* Not two consecutive dots */
|
2014-05-31 12:06:44 +04:00
|
|
|
fail_unless(check_topdomain("abc..defgh", &error));
|
|
|
|
fail_if(strcmp("Consecutive dots", error));
|
2014-05-30 02:18:45 +04:00
|
|
|
|
|
|
|
/* Not end with a dots */
|
2014-05-31 12:06:44 +04:00
|
|
|
fail_unless(check_topdomain("abc.defgh.", &error));
|
|
|
|
fail_if(strcmp("Ends with a dot", error));
|
2014-05-30 02:18:45 +04:00
|
|
|
|
|
|
|
/* No chunk longer than 63 chars */
|
|
|
|
fail_if(check_topdomain("123456789012345678901234567890"
|
2014-05-31 12:06:44 +04:00
|
|
|
"123456789012345678901234567890333.com", &error));
|
|
|
|
fail_unless(check_topdomain("123456789012345678901234567890"
|
|
|
|
"1234567890123456789012345678904444.com", &error));
|
|
|
|
fail_if(strcmp("Too long domain part (> 63)", error));
|
|
|
|
|
2014-05-30 02:18:45 +04:00
|
|
|
fail_if(check_topdomain("abc.123456789012345678901234567890"
|
2014-05-31 12:06:44 +04:00
|
|
|
"123456789012345678901234567890333.com", &error));
|
2014-05-30 02:18:45 +04:00
|
|
|
fail_unless(check_topdomain("abc.123456789012345678901234567890"
|
2014-05-31 12:06:44 +04:00
|
|
|
"1234567890123456789012345678904444.com", &error));
|
|
|
|
fail_if(strcmp("Too long domain part (> 63)", error));
|
|
|
|
|
2014-05-30 02:18:45 +04:00
|
|
|
fail_if(check_topdomain("abc.123456789012345678901234567890"
|
2014-05-31 12:06:44 +04:00
|
|
|
"123456789012345678901234567890333", &error));
|
|
|
|
fail_unless(check_topdomain("abc.123456789012345678901234567890"
|
|
|
|
"1234567890123456789012345678904444", &error));
|
|
|
|
fail_if(strcmp("Too long domain part (> 63)", error));
|
2014-05-30 02:18:45 +04:00
|
|
|
}
|
|
|
|
END_TEST
|
|
|
|
|
2014-02-08 02:08:12 +04:00
|
|
|
START_TEST(test_parse_format_ipv4)
|
|
|
|
{
|
|
|
|
char *host = "192.168.2.10";
|
|
|
|
char *formatted;
|
|
|
|
struct sockaddr_storage addr;
|
|
|
|
struct sockaddr_in *v4addr;
|
|
|
|
int addr_len;
|
|
|
|
|
|
|
|
addr_len = get_addr(host, 53, AF_INET, 0, &addr);
|
|
|
|
fail_unless(addr_len == sizeof(struct sockaddr_in));
|
|
|
|
|
|
|
|
v4addr = (struct sockaddr_in *) &addr;
|
|
|
|
fail_unless(v4addr->sin_addr.s_addr == htonl(0xc0a8020a));
|
|
|
|
fail_unless(v4addr->sin_port == htons(53));
|
|
|
|
|
|
|
|
formatted = format_addr(&addr, addr_len);
|
|
|
|
fail_if(strcmp(host, formatted));
|
|
|
|
}
|
|
|
|
END_TEST
|
|
|
|
|
|
|
|
START_TEST(test_parse_format_ipv4_listen_all)
|
|
|
|
{
|
|
|
|
char *host = "0.0.0.0";
|
|
|
|
char *formatted;
|
|
|
|
struct sockaddr_storage addr;
|
|
|
|
struct sockaddr_in *v4addr;
|
|
|
|
int addr_len;
|
|
|
|
|
|
|
|
addr_len = get_addr(NULL, 53, AF_INET, AI_PASSIVE, &addr);
|
|
|
|
fail_unless(addr_len == sizeof(struct sockaddr_in));
|
|
|
|
|
|
|
|
v4addr = (struct sockaddr_in *) &addr;
|
|
|
|
fail_unless(v4addr->sin_addr.s_addr == htonl(0x00000000));
|
|
|
|
fail_unless(v4addr->sin_port == htons(53));
|
|
|
|
|
|
|
|
formatted = format_addr(&addr, addr_len);
|
|
|
|
fail_if(strcmp(host, formatted));
|
|
|
|
}
|
|
|
|
END_TEST
|
|
|
|
|
2014-02-11 01:34:24 +04:00
|
|
|
START_TEST(test_parse_format_ipv6)
|
|
|
|
{
|
|
|
|
char *host = "2001:0db8:0505:0::123:0abc";
|
|
|
|
char *compact = "2001:db8:505::123:abc";
|
|
|
|
unsigned char v6_bits[] = {
|
|
|
|
0x20, 0x01, 0x0d, 0xb8, 0x05, 0x05, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x01, 0x23, 0x0a, 0xbc,
|
|
|
|
};
|
|
|
|
char *formatted;
|
|
|
|
struct sockaddr_storage addr;
|
|
|
|
struct sockaddr_in6 *v6addr;
|
|
|
|
int addr_len;
|
|
|
|
|
|
|
|
addr_len = get_addr(host, 53, AF_UNSPEC, 0, &addr);
|
|
|
|
fail_unless(addr_len == sizeof(struct sockaddr_in6));
|
|
|
|
|
|
|
|
v6addr = (struct sockaddr_in6 *) &addr;
|
|
|
|
fail_if(memcmp(&v6addr->sin6_addr, v6_bits, sizeof(v6_bits)));
|
|
|
|
fail_unless(v6addr->sin6_port == htons(53));
|
|
|
|
|
|
|
|
formatted = format_addr(&addr, addr_len);
|
|
|
|
fail_if(strcmp(compact, formatted));
|
|
|
|
}
|
|
|
|
END_TEST
|
|
|
|
|
|
|
|
START_TEST(test_parse_format_ipv4_mapped_ipv6)
|
|
|
|
{
|
|
|
|
char *v4mapped = "::FFFF:192.168.2.10";
|
|
|
|
char *host = "192.168.2.10";
|
|
|
|
unsigned char v6_bits[] = {
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0xff, 0xff, 0xc0, 0xa8, 0x02, 0x0a,
|
|
|
|
};
|
|
|
|
char *formatted;
|
|
|
|
struct sockaddr_storage addr;
|
|
|
|
struct sockaddr_in6 *v6addr;
|
|
|
|
int addr_len;
|
|
|
|
|
|
|
|
addr_len = get_addr(v4mapped, 53, AF_INET6, 0, &addr);
|
|
|
|
fail_unless(addr_len == sizeof(struct sockaddr_in6));
|
|
|
|
|
|
|
|
v6addr = (struct sockaddr_in6 *) &addr;
|
|
|
|
fail_if(memcmp(&v6addr->sin6_addr, v6_bits, sizeof(v6_bits)));
|
|
|
|
fail_unless(v6addr->sin6_port == htons(53));
|
|
|
|
|
|
|
|
/* Format as IPv4 address */
|
|
|
|
formatted = format_addr(&addr, addr_len);
|
|
|
|
fail_if(strcmp(host, formatted));
|
|
|
|
}
|
|
|
|
END_TEST
|
|
|
|
|
2014-02-08 02:08:12 +04:00
|
|
|
TCase *
|
|
|
|
test_common_create_tests()
|
|
|
|
{
|
|
|
|
TCase *tc;
|
2014-02-11 01:34:24 +04:00
|
|
|
int sock;
|
2014-02-08 02:08:12 +04:00
|
|
|
|
|
|
|
tc = tcase_create("Common");
|
2014-05-30 02:18:45 +04:00
|
|
|
tcase_add_test(tc, test_topdomain_ok);
|
|
|
|
tcase_add_test(tc, test_topdomain_length);
|
|
|
|
tcase_add_test(tc, test_topdomain_chunks);
|
2014-02-08 02:08:12 +04:00
|
|
|
tcase_add_test(tc, test_parse_format_ipv4);
|
|
|
|
tcase_add_test(tc, test_parse_format_ipv4_listen_all);
|
2014-02-11 01:34:24 +04:00
|
|
|
|
|
|
|
/* Tests require IPv6 support */
|
|
|
|
sock = socket(AF_INET6, SOCK_DGRAM, IPPROTO_UDP);
|
|
|
|
if (sock >= 0) {
|
|
|
|
close(sock);
|
|
|
|
tcase_add_test(tc, test_parse_format_ipv6);
|
|
|
|
tcase_add_test(tc, test_parse_format_ipv4_mapped_ipv6);
|
|
|
|
}
|
2014-02-08 02:08:12 +04:00
|
|
|
return tc;
|
|
|
|
}
|