2006-12-16 03:42:13 +03:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2006 Bjorn Andersson <flex@kryo.se>, Erik Ekman <yarrick@kryo.se>
|
|
|
|
*
|
|
|
|
* 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 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <check.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2007-01-28 16:32:41 +03:00
|
|
|
#include <ctype.h>
|
2006-12-16 03:42:13 +03:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <netinet/in.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <arpa/nameser.h>
|
|
|
|
|
2007-02-04 18:28:24 +03:00
|
|
|
#include "common.h"
|
2006-12-16 03:42:13 +03:00
|
|
|
#include "dns.h"
|
2007-01-28 14:52:36 +03:00
|
|
|
#include "encoding.h"
|
2006-12-16 03:42:13 +03:00
|
|
|
#include "test.h"
|
|
|
|
|
2007-01-28 16:32:41 +03:00
|
|
|
static void dump_packet(char *, size_t);
|
|
|
|
|
2007-01-28 09:11:54 +03:00
|
|
|
static char queryPacket[] =
|
2007-01-28 14:52:36 +03:00
|
|
|
"\x05\x39\x01\x00\x00\x01\x00\x00\x00\x00\x00\x01\x32\x41\x4A\x42\x43"
|
|
|
|
"\x55\x59\x54\x43\x50\x45\x42\x39\x47\x51\x39\x4C\x54\x45\x42\x55\x58"
|
|
|
|
"\x47\x49\x44\x55\x4E\x42\x53\x53\x41\x36\x44\x46\x4F\x4E\x39\x43\x41"
|
|
|
|
"\x5A\x44\x42\x32\x41\x41\x41\x41\x41\x36\x44\x42\x04\x6B\x72\x79\x6F"
|
|
|
|
"\x02\x73\x65\x00\x00\x0A\x00\x01\x00\x00\x29\x10\x00\x00\x00\x80\x00"
|
|
|
|
"\x00\x00";
|
2007-01-28 09:11:54 +03:00
|
|
|
|
|
|
|
static char answerPacket[] =
|
|
|
|
"\x05\x39\x84\x00\x00\x01\x00\x01\x00\x00\x00\x00\x05\x73\x69\x6C\x6C"
|
|
|
|
"\x79\x04\x68\x6F\x73\x74\x02\x6F\x66\x06\x69\x6F\x64\x69\x6E\x65\x04"
|
|
|
|
"\x63\x6F\x64\x65\x04\x6B\x72\x79\x6F\x02\x73\x65\x00\x00\x0A\x00\x01"
|
|
|
|
"\xC0\x0C\x00\x0A\x00\x01\x00\x00\x00\x00\x00\x23\x74\x68\x69\x73\x20"
|
|
|
|
"\x69\x73\x20\x74\x68\x65\x20\x6D\x65\x73\x73\x61\x67\x65\x20\x74\x6F"
|
|
|
|
"\x20\x62\x65\x20\x64\x65\x6C\x69\x76\x65\x72\x65\x64";
|
|
|
|
|
|
|
|
static char *msgData = "this is the message to be delivered";
|
2007-01-28 14:52:36 +03:00
|
|
|
|
|
|
|
static char *queryData = "HELLO this is the test data";
|
|
|
|
static char *recData = "AHELLO this is the test data"; // The A flag is added
|
2007-01-28 09:11:54 +03:00
|
|
|
|
2006-12-16 03:42:13 +03:00
|
|
|
START_TEST(test_encode_hostname)
|
|
|
|
{
|
2007-01-28 06:28:02 +03:00
|
|
|
char out[] = "\x06" "BADGER\x06" "BADGER\x04" "KRYO\x02" "SE\x00";
|
2006-12-16 03:42:13 +03:00
|
|
|
char buf[256];
|
|
|
|
int len;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
len = 256;
|
2007-01-28 06:28:02 +03:00
|
|
|
|
|
|
|
memset(buf, 0, 256);
|
|
|
|
ret = dns_encode_hostname("BADGER.BADGER.KRYO.SE", buf, len);
|
|
|
|
fail_unless(strncmp(buf, out, ret) == 0, "Happy flow failed");
|
|
|
|
}
|
|
|
|
END_TEST
|
2006-12-16 03:42:13 +03:00
|
|
|
|
2007-01-28 06:28:02 +03:00
|
|
|
START_TEST(test_encode_hostname_nodot)
|
|
|
|
{
|
|
|
|
char buf[256];
|
|
|
|
int len;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
len = 256;
|
|
|
|
|
2006-12-16 03:42:13 +03:00
|
|
|
memset(buf, 0, 256);
|
|
|
|
ret = dns_encode_hostname( // More than 63 chars between dots
|
|
|
|
"ABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQRSTUVWXYZ"
|
|
|
|
"ABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQRSTUVWXYZ"
|
|
|
|
, buf, len);
|
|
|
|
fail_unless(ret == -1, NULL);
|
2007-01-28 06:28:02 +03:00
|
|
|
}
|
|
|
|
END_TEST
|
2006-12-16 03:42:13 +03:00
|
|
|
|
2007-01-28 06:28:02 +03:00
|
|
|
START_TEST(test_encode_hostname_toolong)
|
|
|
|
{
|
|
|
|
char buf[256];
|
|
|
|
int len;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
len = 256;
|
|
|
|
|
2006-12-16 03:42:13 +03:00
|
|
|
memset(buf, 0, 256);
|
|
|
|
ret = dns_encode_hostname( // More chars than fits into array
|
|
|
|
"ABCDEFGHIJKLMNOPQRSTUVWXYZ.ABCDEFGHIJKLMNOPQRSTUVWXYZ."
|
|
|
|
"ABCDEFGHIJKLMNOPQRSTUVWXYZ.ABCDEFGHIJKLMNOPQRSTUVWXYZ."
|
|
|
|
"ABCDEFGHIJKLMNOPQRSTUVWXYZ.ABCDEFGHIJKLMNOPQRSTUVWXYZ."
|
|
|
|
"ABCDEFGHIJKLMNOPQRSTUVWXYZ.ABCDEFGHIJKLMNOPQRSTUVWXYZ."
|
|
|
|
"ABCDEFGHIJKLMNOPQRSTUVWXYZ.ABCDEFGHIJKLMNOPQRSTUVWXYZ."
|
|
|
|
"ABCDEFGHIJKLMNOPQRSTUVWXYZ.ABCDEFGHIJKLMNOPQRSTUVWXYZ."
|
|
|
|
, buf, len);
|
|
|
|
|
|
|
|
fail_unless(ret == -1, NULL);
|
|
|
|
fail_unless(strlen(buf) < len, NULL);
|
|
|
|
}
|
|
|
|
END_TEST
|
|
|
|
|
2007-01-28 08:37:13 +03:00
|
|
|
START_TEST(test_encode_query)
|
|
|
|
{
|
|
|
|
char buf[512];
|
2007-01-28 14:52:36 +03:00
|
|
|
char resolv[512];
|
|
|
|
char *host = "kryo.se";
|
2007-01-28 08:37:13 +03:00
|
|
|
struct query q;
|
2007-01-28 14:52:36 +03:00
|
|
|
char *d;
|
2007-01-28 08:37:13 +03:00
|
|
|
int len;
|
2007-01-28 14:52:36 +03:00
|
|
|
int pos;
|
2007-01-28 08:37:13 +03:00
|
|
|
int ret;
|
|
|
|
|
|
|
|
len = sizeof(buf);
|
2007-01-28 14:52:36 +03:00
|
|
|
memset(&buf, 0, sizeof(buf));
|
2007-01-28 16:32:41 +03:00
|
|
|
memset(&resolv, 0, sizeof(resolv));
|
2007-01-28 08:37:13 +03:00
|
|
|
memset(&q, 0, sizeof(struct query));
|
|
|
|
q.type = T_NULL;
|
|
|
|
q.id = 1337;
|
2007-01-28 14:52:36 +03:00
|
|
|
d = resolv;
|
|
|
|
|
|
|
|
encode_data(queryData, strlen(queryData), 100, d, 'A');
|
|
|
|
pos = strlen(resolv);
|
|
|
|
d += pos;
|
|
|
|
if (*d != '.') {
|
|
|
|
*d++ = '.';
|
|
|
|
}
|
|
|
|
strncpy(d, host, strlen(host)+1);
|
|
|
|
ret = dns_encode(buf, len, &q, QR_QUERY, resolv, strlen(resolv));
|
2007-01-28 08:37:13 +03:00
|
|
|
len = sizeof(queryPacket) - 1; // Skip extra null character
|
|
|
|
|
2007-01-28 16:32:41 +03:00
|
|
|
if (strncmp(queryPacket, buf, sizeof(queryPacket)) || ret != len) {
|
|
|
|
printf("\n");
|
|
|
|
dump_packet(queryPacket, len);
|
|
|
|
dump_packet(buf, ret);
|
|
|
|
}
|
2007-01-28 08:37:13 +03:00
|
|
|
fail_unless(strncmp(queryPacket, buf, sizeof(queryPacket)) == 0, "Did not compile expected packet");
|
|
|
|
fail_unless(ret == len, va_str("Bad packet length: %d, expected %d", ret, len));
|
|
|
|
}
|
|
|
|
END_TEST
|
|
|
|
|
2007-01-28 14:52:36 +03:00
|
|
|
START_TEST(test_decode_query)
|
|
|
|
{
|
|
|
|
char buf[512];
|
|
|
|
struct query q;
|
|
|
|
int len;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
memset(&q, 0, sizeof(struct query));
|
|
|
|
memset(&buf, 0, sizeof(buf));
|
|
|
|
q.id = 0;
|
|
|
|
len = sizeof(queryPacket) - 1;
|
|
|
|
|
|
|
|
dns_set_topdomain("kryo.se");
|
|
|
|
ret = dns_decode(buf, sizeof(buf), &q, QR_QUERY, queryPacket, len);
|
|
|
|
|
|
|
|
fail_unless(strncmp(buf, recData, ret) == 0, "Did not extract expected host: '%s'", buf);
|
|
|
|
fail_unless(strlen(buf) == strlen(recData), va_str("Bad host length: %d, expected %d", strlen(q.name), strlen(recData)));
|
|
|
|
}
|
|
|
|
END_TEST
|
|
|
|
|
2007-01-28 08:57:44 +03:00
|
|
|
START_TEST(test_encode_response)
|
|
|
|
{
|
|
|
|
char buf[512];
|
|
|
|
char *host = "silly.host.of.iodine.code.kryo.se";
|
|
|
|
struct query q;
|
|
|
|
int len;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
len = sizeof(buf);
|
2007-01-28 14:52:36 +03:00
|
|
|
memset(&buf, 0, sizeof(buf));
|
2007-01-28 08:57:44 +03:00
|
|
|
memset(&q, 0, sizeof(struct query));
|
|
|
|
strncpy(q.name, host, strlen(host));
|
|
|
|
q.type = T_NULL;
|
|
|
|
q.id = 1337;
|
|
|
|
|
2007-01-28 09:11:54 +03:00
|
|
|
ret = dns_encode(buf, len, &q, QR_ANSWER, msgData, strlen(msgData));
|
2007-01-28 08:57:44 +03:00
|
|
|
len = sizeof(answerPacket) - 1; // Skip extra null character
|
|
|
|
|
|
|
|
fail_unless(strncmp(answerPacket, buf, sizeof(answerPacket)) == 0, "Did not compile expected packet");
|
|
|
|
fail_unless(ret == len, va_str("Bad packet length: %d, expected %d", ret, len));
|
|
|
|
}
|
|
|
|
END_TEST
|
|
|
|
|
2007-01-28 09:11:54 +03:00
|
|
|
START_TEST(test_decode_response)
|
|
|
|
{
|
|
|
|
char buf[512];
|
|
|
|
int len;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
len = sizeof(buf);
|
2007-01-28 14:52:36 +03:00
|
|
|
memset(&buf, 0, sizeof(buf));
|
2007-01-28 09:11:54 +03:00
|
|
|
|
2007-01-28 09:47:51 +03:00
|
|
|
ret = dns_decode(buf, len, NULL, QR_ANSWER, answerPacket, sizeof(answerPacket)-1);
|
2007-01-28 09:11:54 +03:00
|
|
|
fail_unless(strncmp(msgData, buf, sizeof(msgData)) == 0, "Did not extract expected data");
|
|
|
|
fail_unless(ret == strlen(msgData), va_str("Bad data length: %d, expected %d", ret, strlen(msgData)));
|
|
|
|
}
|
|
|
|
END_TEST
|
|
|
|
|
2007-01-28 16:32:41 +03:00
|
|
|
static void
|
|
|
|
dump_packet(char *buf, size_t len)
|
|
|
|
{
|
|
|
|
int pos;
|
|
|
|
|
|
|
|
for (pos = 0; pos < len; pos++) {
|
|
|
|
printf("\\x%02X", (unsigned char) buf[pos]);
|
|
|
|
}
|
|
|
|
printf("\n");
|
|
|
|
for (pos = 0; pos < len; pos++) {
|
|
|
|
if (isalnum((unsigned char) buf[pos])) {
|
|
|
|
printf(" %c ", (unsigned char) buf[pos]);
|
|
|
|
} else {
|
|
|
|
printf(" ");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
printf("\n");
|
|
|
|
}
|
|
|
|
|
2006-12-16 03:42:13 +03:00
|
|
|
TCase *
|
|
|
|
test_dns_create_tests()
|
|
|
|
{
|
|
|
|
TCase *tc;
|
|
|
|
|
|
|
|
tc = tcase_create("Dns");
|
|
|
|
tcase_add_test(tc, test_encode_hostname);
|
2007-01-28 06:28:02 +03:00
|
|
|
tcase_add_test(tc, test_encode_hostname_nodot);
|
|
|
|
tcase_add_test(tc, test_encode_hostname_toolong);
|
2007-01-28 08:37:13 +03:00
|
|
|
tcase_add_test(tc, test_encode_query);
|
2007-01-28 14:52:36 +03:00
|
|
|
tcase_add_test(tc, test_decode_query);
|
2007-01-28 08:57:44 +03:00
|
|
|
tcase_add_test(tc, test_encode_response);
|
2007-01-28 09:11:54 +03:00
|
|
|
tcase_add_test(tc, test_decode_response);
|
2006-12-16 03:42:13 +03:00
|
|
|
|
|
|
|
return tc;
|
|
|
|
}
|