2009-09-27 05:38:41 +02:00
|
|
|
/*
|
|
|
|
|
|
|
|
Copyright (c) 2008, Arvid Norberg
|
|
|
|
All rights reserved.
|
|
|
|
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
|
|
modification, are permitted provided that the following conditions
|
|
|
|
are met:
|
|
|
|
|
|
|
|
* Redistributions of source code must retain the above copyright
|
|
|
|
notice, this list of conditions and the following disclaimer.
|
|
|
|
* Redistributions in binary form must reproduce the above copyright
|
|
|
|
notice, this list of conditions and the following disclaimer in
|
|
|
|
the documentation and/or other materials provided with the distribution.
|
|
|
|
* Neither the name of the author nor the names of its
|
|
|
|
contributors may be used to endorse or promote products derived
|
|
|
|
from this software without specific prior written permission.
|
|
|
|
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
|
|
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
|
|
LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
|
|
CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
|
|
SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
|
|
INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
|
|
CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
|
|
ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
|
|
POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
2010-02-05 09:40:08 +01:00
|
|
|
#ifndef TORRENT_DISABLE_DHT
|
|
|
|
|
2009-09-27 05:38:41 +02:00
|
|
|
#include "libtorrent/session.hpp"
|
2009-09-27 19:41:51 +02:00
|
|
|
#include "libtorrent/kademlia/node.hpp" // for verify_message
|
|
|
|
#include "libtorrent/bencode.hpp"
|
2009-12-15 14:11:07 +01:00
|
|
|
#include <iostream>
|
2009-09-27 05:38:41 +02:00
|
|
|
|
|
|
|
#include "test.hpp"
|
|
|
|
|
2009-09-27 19:41:51 +02:00
|
|
|
using namespace libtorrent;
|
2009-09-27 05:38:41 +02:00
|
|
|
|
2009-09-27 19:41:51 +02:00
|
|
|
int dht_port = 48199;
|
2009-09-27 05:38:41 +02:00
|
|
|
|
2010-11-27 04:09:28 +01:00
|
|
|
void send_dht_msg(datagram_socket& sock, char const* msg, lazy_entry* reply
|
|
|
|
, char const* t = "10", char const* info_hash = 0, char const* name = 0
|
|
|
|
, char const* token = 0, int port = 0)
|
2009-09-27 19:41:51 +02:00
|
|
|
{
|
|
|
|
entry e;
|
|
|
|
e["q"] = msg;
|
|
|
|
e["t"] = t;
|
|
|
|
e["y"] = "q";
|
|
|
|
entry::dictionary_type& a = e["a"].dict();
|
|
|
|
a["id"] = "00000000000000000000";
|
2010-11-27 04:09:28 +01:00
|
|
|
if (info_hash) a["info_hash"] = info_hash;
|
|
|
|
if (name) a["n"] = name;
|
|
|
|
if (token) a["token"] = token;
|
|
|
|
if (port) a["port"] = port;
|
2009-09-27 19:41:51 +02:00
|
|
|
char msg_buf[1500];
|
|
|
|
int size = bencode(msg_buf, e);
|
2009-09-27 05:38:41 +02:00
|
|
|
|
|
|
|
error_code ec;
|
2009-09-27 19:41:51 +02:00
|
|
|
sock.send_to(asio::buffer(msg_buf, size)
|
2009-09-27 05:38:41 +02:00
|
|
|
, udp::endpoint(address::from_string("127.0.0.1"), dht_port), 0, ec);
|
|
|
|
TEST_CHECK(!ec);
|
|
|
|
if (ec) std::cout << ec.message() << std::endl;
|
|
|
|
|
2009-09-27 19:41:51 +02:00
|
|
|
static char inbuf[1500];
|
2009-09-27 05:38:41 +02:00
|
|
|
udp::endpoint ep;
|
2009-09-27 19:41:51 +02:00
|
|
|
size = sock.receive_from(asio::buffer(inbuf, sizeof(inbuf)), ep, 0, ec);
|
2009-09-27 05:38:41 +02:00
|
|
|
TEST_CHECK(!ec);
|
|
|
|
if (ec) std::cout << ec.message() << std::endl;
|
|
|
|
|
2010-10-28 06:01:59 +02:00
|
|
|
int ret = lazy_bdecode(inbuf, inbuf + size, *reply, ec);
|
2009-09-27 05:38:41 +02:00
|
|
|
TEST_CHECK(ret == 0);
|
2009-09-27 19:41:51 +02:00
|
|
|
}
|
2009-09-27 05:38:41 +02:00
|
|
|
|
2009-09-27 19:41:51 +02:00
|
|
|
int test_main()
|
|
|
|
{
|
|
|
|
session ses(fingerprint("LT", 0, 1, 0, 0), std::make_pair(dht_port, 49000));
|
2009-09-27 05:38:41 +02:00
|
|
|
|
2009-09-27 19:41:51 +02:00
|
|
|
// DHT should be running on port 48199 now
|
2009-09-27 05:38:41 +02:00
|
|
|
|
2009-09-27 19:41:51 +02:00
|
|
|
io_service ios;
|
|
|
|
error_code ec;
|
|
|
|
datagram_socket sock(ios);
|
2009-09-27 05:38:41 +02:00
|
|
|
|
2009-09-27 19:41:51 +02:00
|
|
|
sock.open(udp::v4(), ec);
|
|
|
|
TEST_CHECK(!ec);
|
|
|
|
if (ec) std::cout << ec.message() << std::endl;
|
2009-09-27 05:38:41 +02:00
|
|
|
|
2009-09-27 19:41:51 +02:00
|
|
|
lazy_entry response;
|
|
|
|
lazy_entry const* parsed[5];
|
|
|
|
char error_string[200];
|
|
|
|
bool ret;
|
|
|
|
|
|
|
|
// ====== ping ======
|
|
|
|
|
|
|
|
send_dht_msg(sock, "ping", &response, "10");
|
|
|
|
|
|
|
|
dht::key_desc_t pong_desc[] = {
|
|
|
|
{"y", lazy_entry::string_t, 1, 0},
|
|
|
|
{"t", lazy_entry::string_t, 2, 0},
|
|
|
|
};
|
|
|
|
|
2010-11-27 04:09:28 +01:00
|
|
|
fprintf(stderr, "msg: %s\n", print_entry(response).c_str());
|
2009-09-27 19:41:51 +02:00
|
|
|
ret = dht::verify_message(&response, pong_desc, parsed, 2, error_string, sizeof(error_string));
|
|
|
|
TEST_CHECK(ret);
|
|
|
|
if (ret)
|
|
|
|
{
|
|
|
|
TEST_CHECK(parsed[0]->string_value() == "r");
|
|
|
|
TEST_CHECK(parsed[1]->string_value() == "10");
|
|
|
|
}
|
2010-11-27 04:09:28 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
fprintf(stderr, "invalid ping response: %s\n", error_string);
|
|
|
|
}
|
2009-09-27 19:41:51 +02:00
|
|
|
|
|
|
|
// ====== invalid message ======
|
|
|
|
|
|
|
|
send_dht_msg(sock, "find_node", &response, "10");
|
|
|
|
|
|
|
|
dht::key_desc_t err_desc[] = {
|
|
|
|
{"y", lazy_entry::string_t, 1, 0},
|
|
|
|
{"e", lazy_entry::list_t, 0, 0},
|
|
|
|
};
|
|
|
|
|
2010-11-27 04:09:28 +01:00
|
|
|
fprintf(stderr, "msg: %s\n", print_entry(response).c_str());
|
2009-09-27 19:41:51 +02:00
|
|
|
ret = dht::verify_message(&response, err_desc, parsed, 2, error_string, sizeof(error_string));
|
|
|
|
TEST_CHECK(ret);
|
|
|
|
if (ret)
|
|
|
|
{
|
|
|
|
TEST_CHECK(parsed[0]->string_value() == "e");
|
|
|
|
TEST_CHECK(parsed[1]->list_size() >= 2);
|
|
|
|
if (parsed[1]->list_size() >= 2
|
|
|
|
&& parsed[1]->list_at(0)->type() == lazy_entry::int_t
|
|
|
|
&& parsed[1]->list_at(1)->type() == lazy_entry::string_t)
|
|
|
|
{
|
|
|
|
TEST_CHECK(parsed[1]->list_at(1)->string_value() == "missing 'target' key");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
TEST_ERROR("invalid error response");
|
|
|
|
}
|
|
|
|
}
|
2010-11-27 04:09:28 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
fprintf(stderr, "invalid error response: %s\n", error_string);
|
|
|
|
}
|
|
|
|
|
|
|
|
// ====== get_peers ======
|
|
|
|
|
|
|
|
send_dht_msg(sock, "get_peers", &response, "10", "01010101010101010101");
|
|
|
|
|
|
|
|
dht::key_desc_t peer1_desc[] = {
|
|
|
|
{"y", lazy_entry::string_t, 1, 0},
|
|
|
|
{"r", lazy_entry::dict_t, 0, 0},
|
|
|
|
};
|
|
|
|
|
|
|
|
std::string token;
|
|
|
|
fprintf(stderr, "msg: %s\n", print_entry(response).c_str());
|
|
|
|
ret = dht::verify_message(&response, peer1_desc, parsed, 2, error_string, sizeof(error_string));
|
|
|
|
TEST_CHECK(ret);
|
|
|
|
if (ret)
|
|
|
|
{
|
|
|
|
TEST_CHECK(parsed[0]->string_value() == "r");
|
|
|
|
token = parsed[1]->dict_find_string_value("token");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fprintf(stderr, "invalid get_peers response: %s\n", error_string);
|
|
|
|
}
|
|
|
|
|
|
|
|
// ====== announce ======
|
|
|
|
|
|
|
|
send_dht_msg(sock, "announce_peer", &response, "10", "01010101010101010101", "test", token.c_str(), 8080);
|
|
|
|
|
|
|
|
dht::key_desc_t ann_desc[] = {
|
|
|
|
{"y", lazy_entry::string_t, 1, 0},
|
|
|
|
};
|
|
|
|
|
|
|
|
fprintf(stderr, "msg: %s\n", print_entry(response).c_str());
|
|
|
|
ret = dht::verify_message(&response, ann_desc, parsed, 1, error_string, sizeof(error_string));
|
|
|
|
TEST_CHECK(ret);
|
|
|
|
if (ret)
|
|
|
|
{
|
|
|
|
TEST_CHECK(parsed[0]->string_value() == "r");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fprintf(stderr, "invalid announce response: %s\n", error_string);
|
|
|
|
}
|
|
|
|
|
|
|
|
// ====== get_peers ======
|
|
|
|
|
|
|
|
send_dht_msg(sock, "get_peers", &response, "10", "01010101010101010101");
|
|
|
|
|
|
|
|
dht::key_desc_t peer2_desc[] = {
|
|
|
|
{"y", lazy_entry::string_t, 1, 0},
|
|
|
|
{"r", lazy_entry::dict_t, 0, 0},
|
|
|
|
};
|
|
|
|
|
|
|
|
fprintf(stderr, "msg: %s\n", print_entry(response).c_str());
|
|
|
|
ret = dht::verify_message(&response, peer2_desc, parsed, 2, error_string, sizeof(error_string));
|
|
|
|
TEST_CHECK(ret);
|
|
|
|
if (ret)
|
|
|
|
{
|
|
|
|
TEST_CHECK(parsed[0]->string_value() == "r");
|
|
|
|
TEST_EQUAL(parsed[1]->dict_find_string_value("n"), "test");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fprintf(stderr, "invalid get_peers response: %s\n", error_string);
|
|
|
|
}
|
2009-09-27 05:38:41 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-02-05 09:40:08 +01:00
|
|
|
#else
|
|
|
|
|
|
|
|
int test_main()
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|