i2pd/I2PTunnel.cpp

763 lines
25 KiB
C++
Raw Normal View History

2015-01-07 01:51:10 +03:00
#include <cassert>
2015-11-03 17:15:49 +03:00
#include "Base.h"
2014-08-13 23:25:52 +04:00
#include "Log.h"
#include "Destination.h"
2014-10-16 04:52:17 +04:00
#include "ClientContext.h"
2014-08-13 05:14:19 +04:00
#include "I2PTunnel.h"
namespace i2p
{
namespace client
2014-08-13 05:14:19 +04:00
{
2016-07-28 16:25:05 +03:00
/** set standard socket options */
static void I2PTunnelSetSocketOptions(std::shared_ptr<boost::asio::ip::tcp::socket> socket)
{
if (socket && socket->is_open())
{
boost::asio::socket_base::receive_buffer_size option(I2P_TUNNEL_CONNECTION_BUFFER_SIZE);
socket->set_option(option);
}
2016-07-28 16:25:05 +03:00
}
I2PTunnelConnection::I2PTunnelConnection (I2PService * owner, std::shared_ptr<boost::asio::ip::tcp::socket> socket,
2015-03-13 20:29:27 +03:00
std::shared_ptr<const i2p::data::LeaseSet> leaseSet, int port):
I2PServiceHandler(owner), m_Socket (socket), m_RemoteEndpoint (socket->remote_endpoint ()),
m_IsQuiet (true)
2014-08-13 05:14:19 +04:00
{
2015-03-13 20:29:27 +03:00
m_Stream = GetOwner()->GetLocalDestination ()->CreateStream (leaseSet, port);
2014-08-13 05:14:19 +04:00
}
I2PTunnelConnection::I2PTunnelConnection (I2PService * owner,
std::shared_ptr<boost::asio::ip::tcp::socket> socket, std::shared_ptr<i2p::stream::Stream> stream):
I2PServiceHandler(owner), m_Socket (socket), m_Stream (stream),
m_RemoteEndpoint (socket->remote_endpoint ()), m_IsQuiet (true)
{
}
I2PTunnelConnection::I2PTunnelConnection (I2PService * owner, std::shared_ptr<i2p::stream::Stream> stream,
std::shared_ptr<boost::asio::ip::tcp::socket> socket, const boost::asio::ip::tcp::endpoint& target, bool quiet):
I2PServiceHandler(owner), m_Socket (socket), m_Stream (stream),
m_RemoteEndpoint (target), m_IsQuiet (quiet)
2014-08-20 23:03:10 +04:00
{
}
2014-08-13 05:14:19 +04:00
I2PTunnelConnection::~I2PTunnelConnection ()
{
2014-08-14 05:04:23 +04:00
}
2016-07-28 18:16:29 +03:00
void I2PTunnelConnection::I2PConnect (const uint8_t * msg, size_t len)
2014-11-24 06:23:17 +03:00
{
2015-01-07 01:51:10 +03:00
if (m_Stream)
{
if (msg)
m_Stream->Send (msg, len); // connect and send
else
m_Stream->Send (m_Buffer, 0); // connect
}
2014-11-24 06:23:17 +03:00
StreamReceive ();
Receive ();
}
void I2PTunnelConnection::Connect ()
{
I2PTunnelSetSocketOptions(m_Socket);
if (m_Socket) {
#ifdef __linux__
// bind to 127.x.x.x address
// where x.x.x are first three bytes from ident
2016-08-30 00:53:26 +03:00
if (m_RemoteEndpoint.address ().is_v4 () &&
m_RemoteEndpoint.address ().to_v4 ().to_bytes ()[0] == 127)
{
m_Socket->open (boost::asio::ip::tcp::v4 ());
boost::asio::ip::address_v4::bytes_type bytes;
const uint8_t * ident = m_Stream->GetRemoteIdentity ()->GetIdentHash ();
bytes[0] = 127;
memcpy (bytes.data ()+1, ident, 3);
boost::asio::ip::address ourIP = boost::asio::ip::address_v4 (bytes);
m_Socket->bind (boost::asio::ip::tcp::endpoint (ourIP, 0));
}
#endif
m_Socket->async_connect (m_RemoteEndpoint, std::bind (&I2PTunnelConnection::HandleConnect,
2014-11-24 06:23:17 +03:00
shared_from_this (), std::placeholders::_1));
}
2014-11-24 06:23:17 +03:00
}
2014-08-14 05:04:23 +04:00
void I2PTunnelConnection::Terminate ()
{
if (Kill()) return;
2014-10-09 03:44:12 +04:00
if (m_Stream)
{
m_Stream->Close ();
2014-11-23 19:33:58 +03:00
m_Stream.reset ();
2014-10-09 03:44:12 +04:00
}
2014-08-18 03:14:40 +04:00
m_Socket->close ();
Done(shared_from_this ());
2014-08-13 23:25:52 +04:00
}
void I2PTunnelConnection::Receive ()
{
m_Socket->async_read_some (boost::asio::buffer(m_Buffer, I2P_TUNNEL_CONNECTION_BUFFER_SIZE),
2014-11-24 06:23:17 +03:00
std::bind(&I2PTunnelConnection::HandleReceived, shared_from_this (),
std::placeholders::_1, std::placeholders::_2));
2014-08-13 23:25:52 +04:00
}
void I2PTunnelConnection::HandleReceived (const boost::system::error_code& ecode, std::size_t bytes_transferred)
{
if (ecode)
2015-01-03 05:57:37 +03:00
{
2015-12-18 15:12:46 +03:00
LogPrint (eLogError, "I2PTunnel: read error: ", ecode.message ());
2014-10-09 03:44:12 +04:00
if (ecode != boost::asio::error::operation_aborted)
Terminate ();
2014-08-13 23:25:52 +04:00
}
else
2014-08-14 05:04:23 +04:00
{
2014-08-13 23:25:52 +04:00
if (m_Stream)
2015-04-10 01:40:23 +03:00
{
auto s = shared_from_this ();
m_Stream->AsyncSend (m_Buffer, bytes_transferred,
[s](const boost::system::error_code& ecode)
{
if (!ecode)
s->Receive ();
else
s->Terminate ();
});
}
2014-08-13 23:25:52 +04:00
}
}
void I2PTunnelConnection::HandleWrite (const boost::system::error_code& ecode)
{
2014-08-18 03:14:40 +04:00
if (ecode)
{
2015-12-18 15:12:46 +03:00
LogPrint (eLogError, "I2PTunnel: write error: ", ecode.message ());
2014-10-09 03:44:12 +04:00
if (ecode != boost::asio::error::operation_aborted)
Terminate ();
2014-08-18 03:14:40 +04:00
}
else
StreamReceive ();
2014-08-13 23:25:52 +04:00
}
void I2PTunnelConnection::StreamReceive ()
{
if (m_Stream)
2016-02-15 06:10:56 +03:00
{
if (m_Stream->GetStatus () == i2p::stream::eStreamStatusNew ||
m_Stream->GetStatus () == i2p::stream::eStreamStatusOpen) // regular
{
m_Stream->AsyncReceive (boost::asio::buffer (m_StreamBuffer, I2P_TUNNEL_CONNECTION_BUFFER_SIZE),
std::bind (&I2PTunnelConnection::HandleStreamReceive, shared_from_this (),
std::placeholders::_1, std::placeholders::_2),
I2P_TUNNEL_CONNECTION_MAX_IDLE);
}
else // closed by peer
{
// get remaning data
auto len = m_Stream->ReadSome (m_StreamBuffer, I2P_TUNNEL_CONNECTION_BUFFER_SIZE);
if (len > 0) // still some data
Write (m_StreamBuffer, len);
else // no more data
Terminate ();
}
}
2014-08-13 05:14:19 +04:00
}
2014-08-13 23:25:52 +04:00
void I2PTunnelConnection::HandleStreamReceive (const boost::system::error_code& ecode, std::size_t bytes_transferred)
{
if (ecode)
{
2015-12-18 15:12:46 +03:00
LogPrint (eLogError, "I2PTunnel: stream read error: ", ecode.message ());
2014-10-09 03:44:12 +04:00
if (ecode != boost::asio::error::operation_aborted)
2016-02-15 06:10:56 +03:00
{
if (bytes_transferred > 0)
Write (m_StreamBuffer, bytes_transferred); // postpone termination
else
Terminate ();
2016-10-20 22:20:08 +03:00
}
else
Terminate ();
2014-08-13 23:25:52 +04:00
}
else
2015-06-02 20:03:22 +03:00
Write (m_StreamBuffer, bytes_transferred);
}
void I2PTunnelConnection::Write (const uint8_t * buf, size_t len)
{
2016-02-03 03:24:49 +03:00
boost::asio::async_write (*m_Socket, boost::asio::buffer (buf, len), boost::asio::transfer_all (),
2015-06-02 20:03:22 +03:00
std::bind (&I2PTunnelConnection::HandleWrite, shared_from_this (), std::placeholders::_1));
2014-08-13 23:25:52 +04:00
}
2014-08-20 23:03:10 +04:00
void I2PTunnelConnection::HandleConnect (const boost::system::error_code& ecode)
{
if (ecode)
{
2015-12-18 15:12:46 +03:00
LogPrint (eLogError, "I2PTunnel: connect error: ", ecode.message ());
2014-11-24 06:23:17 +03:00
Terminate ();
2014-08-20 23:03:10 +04:00
}
else
{
2015-12-18 15:12:46 +03:00
LogPrint (eLogDebug, "I2PTunnel: connected");
if (m_IsQuiet)
StreamReceive ();
else
{
// send destination first like received from I2P
2015-11-03 17:15:49 +03:00
std::string dest = m_Stream->GetRemoteIdentity ()->ToBase64 ();
dest += "\n";
memcpy (m_StreamBuffer, dest.c_str (), dest.size ());
HandleStreamReceive (boost::system::error_code (), dest.size ());
}
2014-08-20 23:03:10 +04:00
Receive ();
}
}
2015-06-02 20:03:22 +03:00
I2PTunnelConnectionHTTP::I2PTunnelConnectionHTTP (I2PService * owner, std::shared_ptr<i2p::stream::Stream> stream,
std::shared_ptr<boost::asio::ip::tcp::socket> socket,
const boost::asio::ip::tcp::endpoint& target, const std::string& host):
2016-01-11 21:48:18 +03:00
I2PTunnelConnection (owner, stream, socket, target), m_Host (host), m_HeaderSent (false), m_From (stream->GetRemoteIdentity ())
2015-06-02 20:03:22 +03:00
{
}
void I2PTunnelConnectionHTTP::Write (const uint8_t * buf, size_t len)
{
if (m_HeaderSent)
I2PTunnelConnection::Write (buf, len);
else
2015-06-03 19:30:15 +03:00
{
m_InHeader.clear ();
m_InHeader.write ((const char *)buf, len);
std::string line;
bool endOfHeader = false;
while (!endOfHeader)
{
std::getline(m_InHeader, line);
if (!m_InHeader.fail ())
{
if (line == "\r") endOfHeader = true;
2016-02-03 03:24:49 +03:00
else
{
if (line.find ("Host:") != std::string::npos)
m_OutHeader << "Host: " << m_Host << "\r\n";
else
m_OutHeader << line << "\n";
}
2015-06-03 19:30:15 +03:00
}
else
break;
}
2016-01-11 21:48:18 +03:00
// add X-I2P fields
if (m_From)
{
2016-01-19 17:36:56 +03:00
m_OutHeader << X_I2P_DEST_B32 << ": " << context.GetAddressBook ().ToAddress(m_From->GetIdentHash ()) << "\r\n";
2016-01-11 21:48:18 +03:00
m_OutHeader << X_I2P_DEST_HASH << ": " << m_From->GetIdentHash ().ToBase64 () << "\r\n";
2016-01-19 17:36:56 +03:00
m_OutHeader << X_I2P_DEST_B64 << ": " << m_From->ToBase64 () << "\r\n";
2016-01-11 21:48:18 +03:00
}
2015-06-03 19:30:15 +03:00
if (endOfHeader)
{
2016-02-03 03:24:49 +03:00
m_OutHeader << "\r\n"; // end of header
2016-02-03 06:00:51 +03:00
m_OutHeader << m_InHeader.str ().substr (m_InHeader.tellg ()); // data right after header
2015-06-03 19:30:15 +03:00
m_HeaderSent = true;
I2PTunnelConnection::Write ((uint8_t *)m_OutHeader.str ().c_str (), m_OutHeader.str ().length ());
}
}
}
2016-02-22 22:33:21 +03:00
I2PTunnelConnectionIRC::I2PTunnelConnectionIRC (I2PService * owner, std::shared_ptr<i2p::stream::Stream> stream,
std::shared_ptr<boost::asio::ip::tcp::socket> socket,
2016-03-04 09:37:38 +03:00
const boost::asio::ip::tcp::endpoint& target, const std::string& webircpass):
I2PTunnelConnection (owner, stream, socket, target), m_From (stream->GetRemoteIdentity ()),
2016-03-04 17:26:28 +03:00
m_NeedsWebIrc (webircpass.length() ? true : false), m_WebircPass (webircpass)
2016-02-22 22:33:21 +03:00
{
}
void I2PTunnelConnectionIRC::Write (const uint8_t * buf, size_t len)
{
m_OutPacket.str ("");
if (m_NeedsWebIrc)
{
2016-03-04 17:26:28 +03:00
m_NeedsWebIrc = false;
m_OutPacket << "WEBIRC " << m_WebircPass << " cgiirc " << context.GetAddressBook ().ToAddress (m_From->GetIdentHash ()) << " 127.0.0.1\n";
2016-03-04 09:37:38 +03:00
}
2016-02-28 16:03:12 +03:00
m_InPacket.clear ();
2016-02-22 22:33:21 +03:00
m_InPacket.write ((const char *)buf, len);
while (!m_InPacket.eof () && !m_InPacket.fail ())
{
std::string line;
2016-02-23 07:48:46 +03:00
std::getline (m_InPacket, line);
if (line.length () == 0 && m_InPacket.eof ())
2016-02-29 00:15:29 +03:00
m_InPacket.str ("");
2016-02-23 07:48:46 +03:00
auto pos = line.find ("USER");
if (!pos) // start of line
2016-02-23 07:48:46 +03:00
{
pos = line.find (" ");
pos++;
pos = line.find (" ", pos);
pos++;
auto nextpos = line.find (" ", pos);
m_OutPacket << line.substr (0, pos);
m_OutPacket << context.GetAddressBook ().ToAddress (m_From->GetIdentHash ());
m_OutPacket << line.substr (nextpos) << '\n';
}
else
2016-02-23 07:48:46 +03:00
m_OutPacket << line << '\n';
2016-02-22 22:33:21 +03:00
}
I2PTunnelConnection::Write ((uint8_t *)m_OutPacket.str ().c_str (), m_OutPacket.str ().length ());
}
/* This handler tries to stablish a connection with the desired server and dies if it fails to do so */
class I2PClientTunnelHandler: public I2PServiceHandler, public std::enable_shared_from_this<I2PClientTunnelHandler>
{
public:
I2PClientTunnelHandler (I2PClientTunnel * parent, i2p::data::IdentHash destination,
int destinationPort, std::shared_ptr<boost::asio::ip::tcp::socket> socket):
2015-03-13 20:29:27 +03:00
I2PServiceHandler(parent), m_DestinationIdentHash(destination),
m_DestinationPort (destinationPort), m_Socket(socket) {};
void Handle();
void Terminate();
private:
void HandleStreamRequestComplete (std::shared_ptr<i2p::stream::Stream> stream);
i2p::data::IdentHash m_DestinationIdentHash;
2015-03-13 20:29:27 +03:00
int m_DestinationPort;
std::shared_ptr<boost::asio::ip::tcp::socket> m_Socket;
};
void I2PClientTunnelHandler::Handle()
{
2015-03-13 20:29:27 +03:00
GetOwner()->GetLocalDestination ()->CreateStream (
std::bind (&I2PClientTunnelHandler::HandleStreamRequestComplete, shared_from_this(), std::placeholders::_1),
m_DestinationIdentHash, m_DestinationPort);
}
void I2PClientTunnelHandler::HandleStreamRequestComplete (std::shared_ptr<i2p::stream::Stream> stream)
{
if (stream)
{
if (Kill()) return;
2015-12-18 15:12:46 +03:00
LogPrint (eLogDebug, "I2PTunnel: new connection");
auto connection = std::make_shared<I2PTunnelConnection>(GetOwner(), m_Socket, stream);
GetOwner()->AddHandler (connection);
connection->I2PConnect ();
Done(shared_from_this());
}
else
{
2015-12-18 15:12:46 +03:00
LogPrint (eLogError, "I2PTunnel: Client Tunnel Issue when creating the stream, check the previous warnings for more info.");
Terminate();
}
}
void I2PClientTunnelHandler::Terminate()
{
if (Kill()) return;
if (m_Socket)
{
m_Socket->close();
m_Socket = nullptr;
}
Done(shared_from_this());
}
2016-01-14 04:21:53 +03:00
I2PClientTunnel::I2PClientTunnel (const std::string& name, const std::string& destination,
const std::string& address, int port, std::shared_ptr<ClientDestination> localDestination, int destinationPort):
TCPIPAcceptor (address, port, localDestination), m_Name (name), m_Destination (destination),
m_DestinationIdentHash (nullptr), m_DestinationPort (destinationPort)
{
}
2014-08-13 05:14:19 +04:00
2014-08-13 23:25:52 +04:00
void I2PClientTunnel::Start ()
{
TCPIPAcceptor::Start ();
GetIdentHash();
2014-08-13 23:25:52 +04:00
}
void I2PClientTunnel::Stop ()
{
2015-01-08 05:49:35 +03:00
TCPIPAcceptor::Stop();
2015-01-03 04:07:55 +03:00
auto *originalIdentHash = m_DestinationIdentHash;
2014-08-13 23:25:52 +04:00
m_DestinationIdentHash = nullptr;
2015-01-03 04:07:55 +03:00
delete originalIdentHash;
2014-08-13 23:25:52 +04:00
}
/* HACK: maybe we should create a caching IdentHash provider in AddressBook */
const i2p::data::IdentHash * I2PClientTunnel::GetIdentHash ()
{
if (!m_DestinationIdentHash)
{
i2p::data::IdentHash identHash;
if (i2p::client::context.GetAddressBook ().GetIdentHash (m_Destination, identHash))
m_DestinationIdentHash = new i2p::data::IdentHash (identHash);
else
2015-12-18 15:12:46 +03:00
LogPrint (eLogWarning, "I2PTunnel: Remote destination ", m_Destination, " not found");
}
return m_DestinationIdentHash;
}
std::shared_ptr<I2PServiceHandler> I2PClientTunnel::CreateHandler(std::shared_ptr<boost::asio::ip::tcp::socket> socket)
2014-08-13 05:14:19 +04:00
{
2015-01-08 05:49:35 +03:00
const i2p::data::IdentHash *identHash = GetIdentHash();
if (identHash)
2015-03-13 20:29:27 +03:00
return std::make_shared<I2PClientTunnelHandler>(this, *identHash, m_DestinationPort, socket);
2014-08-13 05:14:19 +04:00
else
2015-01-08 05:49:35 +03:00
return nullptr;
2014-10-15 20:07:06 +04:00
}
2016-01-14 04:21:53 +03:00
I2PServerTunnel::I2PServerTunnel (const std::string& name, const std::string& address,
int port, std::shared_ptr<ClientDestination> localDestination, int inport, bool gzip):
2016-01-14 04:21:53 +03:00
I2PService (localDestination), m_Name (name), m_Address (address), m_Port (port), m_IsAccessList (false)
2014-08-20 23:03:10 +04:00
{
m_PortDestination = localDestination->CreateStreamingDestination (inport > 0 ? inport : port, gzip);
2014-08-20 23:03:10 +04:00
}
2016-07-28 18:16:29 +03:00
2014-08-20 23:03:10 +04:00
void I2PServerTunnel::Start ()
{
m_Endpoint.port (m_Port);
boost::system::error_code ec;
auto addr = boost::asio::ip::address::from_string (m_Address, ec);
if (!ec)
{
m_Endpoint.address (addr);
Accept ();
}
else
{
auto resolver = std::make_shared<boost::asio::ip::tcp::resolver>(GetService ());
resolver->async_resolve (boost::asio::ip::tcp::resolver::query (m_Address, ""),
std::bind (&I2PServerTunnel::HandleResolve, this,
std::placeholders::_1, std::placeholders::_2, resolver));
}
2014-08-20 23:03:10 +04:00
}
void I2PServerTunnel::Stop ()
{
ClearHandlers ();
2014-08-20 23:03:10 +04:00
}
void I2PServerTunnel::HandleResolve (const boost::system::error_code& ecode, boost::asio::ip::tcp::resolver::iterator it,
std::shared_ptr<boost::asio::ip::tcp::resolver> resolver)
{
if (!ecode)
{
auto addr = (*it).endpoint ().address ();
2015-12-18 15:12:46 +03:00
LogPrint (eLogInfo, "I2PTunnel: server tunnel ", (*it).host_name (), " has been resolved to ", addr);
m_Endpoint.address (addr);
Accept ();
}
else
2015-12-18 15:12:46 +03:00
LogPrint (eLogError, "I2PTunnel: Unable to resolve server tunnel address: ", ecode.message ());
}
2015-03-16 21:52:42 +03:00
void I2PServerTunnel::SetAccessList (const std::set<i2p::data::IdentHash>& accessList)
{
m_AccessList = accessList;
m_IsAccessList = true;
}
2014-08-20 23:03:10 +04:00
void I2PServerTunnel::Accept ()
{
if (m_PortDestination)
m_PortDestination->SetAcceptor (std::bind (&I2PServerTunnel::HandleAccept, this, std::placeholders::_1));
auto localDestination = GetLocalDestination ();
if (localDestination)
{
if (!localDestination->IsAcceptingStreams ()) // set it as default if not set yet
localDestination->AcceptStreams (std::bind (&I2PServerTunnel::HandleAccept, this, std::placeholders::_1));
}
else
2015-12-18 15:12:46 +03:00
LogPrint (eLogError, "I2PTunnel: Local destination not set for server tunnel");
2014-08-20 23:03:10 +04:00
}
2014-11-23 19:33:58 +03:00
void I2PServerTunnel::HandleAccept (std::shared_ptr<i2p::stream::Stream> stream)
2014-08-20 23:03:10 +04:00
{
if (stream)
2014-11-24 06:23:17 +03:00
{
2015-03-16 21:52:42 +03:00
if (m_IsAccessList)
{
2015-11-03 17:15:49 +03:00
if (!m_AccessList.count (stream->GetRemoteIdentity ()->GetIdentHash ()))
2015-03-16 21:52:42 +03:00
{
2015-12-18 15:12:46 +03:00
LogPrint (eLogWarning, "I2PTunnel: Address ", stream->GetRemoteIdentity ()->GetIdentHash ().ToBase32 (), " is not in white list. Incoming connection dropped");
2015-03-16 21:52:42 +03:00
stream->Close ();
return;
}
}
2015-06-02 20:03:22 +03:00
CreateI2PConnection (stream);
2014-11-24 06:23:17 +03:00
}
2014-08-20 23:03:10 +04:00
}
2015-05-20 23:00:09 +03:00
2015-06-02 20:03:22 +03:00
void I2PServerTunnel::CreateI2PConnection (std::shared_ptr<i2p::stream::Stream> stream)
{
auto conn = std::make_shared<I2PTunnelConnection> (this, stream, std::make_shared<boost::asio::ip::tcp::socket> (GetService ()), GetEndpoint ());
AddHandler (conn);
conn->Connect ();
}
2016-01-14 04:21:53 +03:00
I2PServerTunnelHTTP::I2PServerTunnelHTTP (const std::string& name, const std::string& address,
2016-02-26 04:32:05 +03:00
int port, std::shared_ptr<ClientDestination> localDestination,
const std::string& host, int inport, bool gzip):
I2PServerTunnel (name, address, port, localDestination, inport, gzip),
2016-02-26 04:32:05 +03:00
m_Host (host.length () > 0 ? host : address)
2015-06-02 20:03:22 +03:00
{
}
void I2PServerTunnelHTTP::CreateI2PConnection (std::shared_ptr<i2p::stream::Stream> stream)
2015-05-20 23:00:09 +03:00
{
2016-02-26 04:32:05 +03:00
auto conn = std::make_shared<I2PTunnelConnectionHTTP> (this, stream,
std::make_shared<boost::asio::ip::tcp::socket> (GetService ()), GetEndpoint (), m_Host);
2015-06-02 20:03:22 +03:00
AddHandler (conn);
conn->Connect ();
2015-05-20 23:00:09 +03:00
}
2016-02-22 22:33:21 +03:00
I2PServerTunnelIRC::I2PServerTunnelIRC (const std::string& name, const std::string& address,
2016-03-04 09:37:38 +03:00
int port, std::shared_ptr<ClientDestination> localDestination,
const std::string& webircpass, int inport, bool gzip):
I2PServerTunnel (name, address, port, localDestination, inport, gzip),
m_WebircPass (webircpass)
2016-02-22 22:33:21 +03:00
{
}
void I2PServerTunnelIRC::CreateI2PConnection (std::shared_ptr<i2p::stream::Stream> stream)
{
2016-03-04 09:37:38 +03:00
auto conn = std::make_shared<I2PTunnelConnectionIRC> (this, stream, std::make_shared<boost::asio::ip::tcp::socket> (GetService ()), GetEndpoint (), this->m_WebircPass);
2016-02-23 07:48:46 +03:00
AddHandler (conn);
conn->Connect ();
2016-02-22 22:33:21 +03:00
}
2016-03-04 09:37:38 +03:00
2016-08-21 22:02:17 +03:00
void I2PUDPServerTunnel::HandleRecvFromI2P(const i2p::data::IdentityEx& from, uint16_t fromPort, uint16_t toPort, const uint8_t * buf, size_t len)
{
std::lock_guard<std::mutex> lock(m_SessionsMutex);
2016-08-22 20:54:00 +03:00
auto session = ObtainUDPSession(from, toPort, fromPort);
2016-09-03 16:38:53 +03:00
session->IPSocket.send_to(boost::asio::buffer(buf, len), m_RemoteEndpoint);
2016-08-22 20:54:00 +03:00
session->LastActivity = i2p::util::GetMillisecondsSinceEpoch();
2016-08-21 22:02:17 +03:00
}
void I2PUDPServerTunnel::ExpireStale(const uint64_t delta) {
std::lock_guard<std::mutex> lock(m_SessionsMutex);
uint64_t now = i2p::util::GetMillisecondsSinceEpoch();
2016-08-22 20:54:00 +03:00
std::remove_if(m_Sessions.begin(), m_Sessions.end(), [now, delta](const UDPSession * u) -> bool {
return now - u->LastActivity >= delta;
2016-08-21 22:02:17 +03:00
});
}
2016-08-22 20:54:00 +03:00
UDPSession * I2PUDPServerTunnel::ObtainUDPSession(const i2p::data::IdentityEx& from, uint16_t localPort, uint16_t remotePort)
2016-08-21 22:02:17 +03:00
{
auto ih = from.GetIdentHash();
2016-09-03 16:38:53 +03:00
for ( UDPSession * s : m_Sessions )
{
if ( s->Identity == ih)
{
/** found existing session */
2016-08-23 01:29:12 +03:00
LogPrint(eLogDebug, "UDPServer: found session ", s->IPSocket.local_endpoint(), " ", ih.ToBase32());
2016-08-21 22:02:17 +03:00
return s;
}
}
2016-09-03 16:38:53 +03:00
/** create new udp session */
boost::asio::ip::udp::endpoint ep(m_LocalAddress, 0);
2016-10-09 17:55:55 +03:00
m_Sessions.push_back(new UDPSession(ep, m_LocalDest, m_RemoteEndpoint, &ih, localPort, remotePort));
2016-08-22 20:54:00 +03:00
return m_Sessions.back();
2016-08-21 22:02:17 +03:00
}
2016-09-03 16:38:53 +03:00
UDPSession::UDPSession(boost::asio::ip::udp::endpoint localEndpoint,
const std::shared_ptr<i2p::client::ClientDestination> & localDestination,
2016-10-09 17:55:55 +03:00
boost::asio::ip::udp::endpoint endpoint, const i2p::data::IdentHash * to,
2016-09-03 16:38:53 +03:00
uint16_t ourPort, uint16_t theirPort) :
2016-08-22 20:54:00 +03:00
m_Destination(localDestination->GetDatagramDestination()),
m_Service(localDestination->GetService()),
2016-09-04 00:53:46 +03:00
IPSocket(localDestination->GetService(), localEndpoint),
SendEndpoint(endpoint),
LastActivity(i2p::util::GetMillisecondsSinceEpoch()),
LocalPort(ourPort),
RemotePort(theirPort)
{
2016-10-09 17:55:55 +03:00
memcpy(Identity, to->data(), 32);
2016-09-04 00:53:46 +03:00
Receive();
}
2016-08-21 22:02:17 +03:00
2016-09-04 00:53:46 +03:00
void UDPSession::Receive() {
LogPrint(eLogDebug, "UDPSession: Receive");
IPSocket.async_receive_from(boost::asio::buffer(m_Buffer, I2P_UDP_MAX_MTU),
FromEndpoint, std::bind(&UDPSession::HandleReceived, this, std::placeholders::_1, std::placeholders::_2));
}
void UDPSession::HandleReceived(const boost::system::error_code & ecode, std::size_t len)
{
if(!ecode)
{
LogPrint(eLogDebug, "UDPSession: forward ", len, "B from ", FromEndpoint);
LastActivity = i2p::util::GetMillisecondsSinceEpoch();
2016-10-09 17:55:55 +03:00
m_Destination->SendDatagramTo(m_Buffer, len, Identity, 0, 0);
2016-09-04 00:53:46 +03:00
Receive();
} else {
LogPrint(eLogError, "UDPSession: ", ecode.message());
}
}
2016-08-22 20:54:00 +03:00
2016-09-04 00:53:46 +03:00
I2PUDPServerTunnel::I2PUDPServerTunnel(const std::string & name, std::shared_ptr<i2p::client::ClientDestination> localDestination,
const boost::asio::ip::address& localAddress, boost::asio::ip::udp::endpoint forwardTo, uint16_t port) :
m_Name(name),
LocalPort(port),
m_LocalAddress(localAddress),
m_RemoteEndpoint(forwardTo)
{
m_LocalDest = localDestination;
m_LocalDest->Start();
auto dgram = m_LocalDest->CreateDatagramDestination();
dgram->SetReceiver(std::bind(&I2PUDPServerTunnel::HandleRecvFromI2P, this, std::placeholders::_1, std::placeholders::_2, std::placeholders::_3, std::placeholders::_4, std::placeholders::_5));
}
2016-08-21 22:02:17 +03:00
2016-09-04 00:53:46 +03:00
I2PUDPServerTunnel::~I2PUDPServerTunnel()
{
auto dgram = m_LocalDest->GetDatagramDestination();
if (dgram) dgram->ResetReceiver();
LogPrint(eLogInfo, "UDPServer: done");
}
2016-08-21 22:02:17 +03:00
2016-09-04 00:53:46 +03:00
void I2PUDPServerTunnel::Start() {
m_LocalDest->Start();
}
2016-09-03 20:58:34 +03:00
2016-09-04 00:53:46 +03:00
std::vector<std::shared_ptr<DatagramSessionInfo> > I2PUDPServerTunnel::GetSessions()
2016-09-03 20:58:34 +03:00
{
2016-09-04 00:53:46 +03:00
std::vector<std::shared_ptr<DatagramSessionInfo> > sessions;
2016-09-03 20:58:34 +03:00
std::lock_guard<std::mutex> lock(m_SessionsMutex);
for ( UDPSession * s : m_Sessions )
{
if (!s->m_Destination) continue;
auto info = s->m_Destination->GetInfoForRemote(s->Identity);
if(!info) continue;
2016-09-04 00:53:46 +03:00
auto sinfo = std::make_shared<DatagramSessionInfo>();
sinfo->Name = m_Name;
sinfo->LocalIdent = std::make_shared<i2p::data::IdentHash>(m_LocalDest->GetIdentHash().data());
sinfo->RemoteIdent = std::make_shared<i2p::data::IdentHash>(s->Identity.data());
sinfo->CurrentIBGW = info->IBGW;
sinfo->CurrentOBEP = info->OBEP;
sessions.push_back(sinfo);
2016-09-03 20:58:34 +03:00
}
return sessions;
}
2016-09-04 00:53:46 +03:00
I2PUDPClientTunnel::I2PUDPClientTunnel(const std::string & name, const std::string &remoteDest,
boost::asio::ip::udp::endpoint localEndpoint,
std::shared_ptr<i2p::client::ClientDestination> localDestination,
uint16_t remotePort) :
m_Name(name),
m_Session(nullptr),
m_RemoteDest(remoteDest),
m_LocalDest(localDestination),
m_LocalEndpoint(localEndpoint),
m_RemoteIdent(nullptr),
m_ResolveThread(nullptr),
LocalPort(localEndpoint.port()),
RemotePort(remotePort),
m_cancel_resolve(false)
{
auto dgram = m_LocalDest->CreateDatagramDestination();
dgram->SetReceiver(std::bind(&I2PUDPClientTunnel::HandleRecvFromI2P, this,
std::placeholders::_1, std::placeholders::_2,
std::placeholders::_3, std::placeholders::_4,
std::placeholders::_5));
}
2016-08-21 22:02:17 +03:00
2016-09-04 00:53:46 +03:00
void I2PUDPClientTunnel::Start() {
m_LocalDest->Start();
if (m_ResolveThread == nullptr)
m_ResolveThread = new std::thread(std::bind(&I2PUDPClientTunnel::TryResolving, this));
}
2016-08-21 22:02:17 +03:00
2016-09-04 00:53:46 +03:00
std::vector<std::shared_ptr<DatagramSessionInfo> > I2PUDPClientTunnel::GetSessions()
2016-09-03 20:58:34 +03:00
{
2016-09-04 00:53:46 +03:00
std::vector<std::shared_ptr<DatagramSessionInfo> > infos;
2016-09-03 20:58:34 +03:00
if(m_Session && m_LocalDest)
{
auto s = m_Session;
if (s->m_Destination)
{
auto info = m_Session->m_Destination->GetInfoForRemote(s->Identity);
2016-09-03 23:16:16 +03:00
if(info)
2016-09-03 20:58:34 +03:00
{
2016-09-04 00:53:46 +03:00
auto sinfo = std::make_shared<DatagramSessionInfo>();
sinfo->Name = m_Name;
sinfo->LocalIdent = std::make_shared<i2p::data::IdentHash>(m_LocalDest->GetIdentHash().data());
sinfo->RemoteIdent = std::make_shared<i2p::data::IdentHash>(s->Identity.data());
sinfo->CurrentIBGW = info->IBGW;
sinfo->CurrentOBEP = info->OBEP;
infos.push_back(sinfo);
2016-09-03 20:58:34 +03:00
}
}
}
return infos;
}
2016-09-04 00:53:46 +03:00
void I2PUDPClientTunnel::TryResolving() {
LogPrint(eLogInfo, "UDP Tunnel: Trying to resolve ", m_RemoteDest);
m_RemoteIdent = new i2p::data::IdentHash;
m_RemoteIdent->Fill(0);
2016-08-22 05:11:41 +03:00
2016-09-04 00:53:46 +03:00
while(!context.GetAddressBook().GetIdentHash(m_RemoteDest, *m_RemoteIdent) && !m_cancel_resolve)
{
LogPrint(eLogWarning, "UDP Tunnel: failed to lookup ", m_RemoteDest);
std::this_thread::sleep_for(std::chrono::seconds(1));
}
if(m_cancel_resolve)
{
LogPrint(eLogError, "UDP Tunnel: lookup of ", m_RemoteDest, " was cancelled");
return;
}
LogPrint(eLogInfo, "UDP Tunnel: resolved ", m_RemoteDest, " to ", m_RemoteIdent->ToBase32());
// delete existing session
if(m_Session) delete m_Session;
boost::asio::ip::udp::endpoint ep(boost::asio::ip::address::from_string("127.0.0.1"), 0);
2016-10-09 17:55:55 +03:00
m_Session = new UDPSession(m_LocalEndpoint, m_LocalDest, ep, m_RemoteIdent, LocalPort, RemotePort);
2016-09-04 00:53:46 +03:00
}
2016-08-21 22:02:17 +03:00
void I2PUDPClientTunnel::HandleRecvFromI2P(const i2p::data::IdentityEx& from, uint16_t fromPort, uint16_t toPort, const uint8_t * buf, size_t len)
2016-09-04 00:53:46 +03:00
{
if(m_RemoteIdent && from.GetIdentHash() == *m_RemoteIdent)
{
// address match
if(m_Session)
{
// tell session
LogPrint(eLogDebug, "UDP Client: got ", len, "B from ", from.GetIdentHash().ToBase32());
m_Session->IPSocket.send_to(boost::asio::buffer(buf, len), m_Session->FromEndpoint);
}
else
LogPrint(eLogWarning, "UDP Client: no session");
}
else
LogPrint(eLogWarning, "UDP Client: unwarrented traffic from ", from.GetIdentHash().ToBase32());
}
2016-09-03 16:38:53 +03:00
2016-09-04 00:53:46 +03:00
I2PUDPClientTunnel::~I2PUDPClientTunnel() {
auto dgram = m_LocalDest->GetDatagramDestination();
if (dgram) dgram->ResetReceiver();
2016-09-03 16:38:53 +03:00
2016-09-04 00:53:46 +03:00
if (m_Session) delete m_Session;
m_cancel_resolve = true;
2016-09-03 16:38:53 +03:00
2016-09-04 00:53:46 +03:00
if(m_ResolveThread)
{
m_ResolveThread->join();
delete m_ResolveThread;
m_ResolveThread = nullptr;
}
if (m_RemoteIdent) delete m_RemoteIdent;
}
2016-09-03 16:38:53 +03:00
}
}