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