i2pd/I2PTunnel.h

281 lines
9.2 KiB
C
Raw Normal View History

2014-08-13 01:14:19 +00:00
#ifndef I2PTUNNEL_H__
#define I2PTUNNEL_H__
#include <inttypes.h>
#include <string>
2014-08-13 19:25:52 +00:00
#include <set>
2014-11-23 16:33:58 +00:00
#include <memory>
#include <sstream>
2014-08-13 01:14:19 +00:00
#include <boost/asio.hpp>
#include "Identity.h"
#include "Destination.h"
2016-08-21 19:02:17 +00:00
#include "Datagram.h"
2014-08-13 01:14:19 +00:00
#include "Streaming.h"
#include "I2PService.h"
2014-08-13 01:14:19 +00:00
namespace i2p
{
namespace client
2014-08-13 01:14:19 +00:00
{
2014-08-13 19:25:52 +00:00
const size_t I2P_TUNNEL_CONNECTION_BUFFER_SIZE = 8192;
const int I2P_TUNNEL_CONNECTION_MAX_IDLE = 3600; // in seconds
2014-10-15 16:07:06 +00:00
const int I2P_TUNNEL_DESTINATION_REQUEST_TIMEOUT = 10; // in seconds
2016-01-11 18:48:18 +00:00
// for HTTP tunnels
const char X_I2P_DEST_HASH[] = "X-I2P-DestHash"; // hash in base64
2016-01-19 14:36:56 +00:00
const char X_I2P_DEST_B64[] = "X-I2P-DestB64"; // full address in base64
const char X_I2P_DEST_B32[] = "X-I2P-DestB32"; // .b32.i2p address
class I2PTunnelConnection: public I2PServiceHandler, public std::enable_shared_from_this<I2PTunnelConnection>
2014-08-13 01:14:19 +00:00
{
public:
I2PTunnelConnection (I2PService * owner, std::shared_ptr<boost::asio::ip::tcp::socket> socket,
2015-03-13 17:29:27 +00:00
std::shared_ptr<const i2p::data::LeaseSet> leaseSet, int port = 0); // to I2P
I2PTunnelConnection (I2PService * owner, std::shared_ptr<boost::asio::ip::tcp::socket> socket,
2015-06-02 17:03:22 +00:00
std::shared_ptr<i2p::stream::Stream> stream); // to I2P using simplified API
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 = true); // from I2P
2014-08-13 01:14:19 +00:00
~I2PTunnelConnection ();
void I2PConnect (const uint8_t * msg = nullptr, size_t len = 0);
2014-11-24 03:23:17 +00:00
void Connect ();
2015-06-02 17:03:22 +00:00
protected:
2014-08-13 01:14:19 +00:00
2014-08-13 19:25:52 +00:00
void Terminate ();
void Receive ();
void HandleReceived (const boost::system::error_code& ecode, std::size_t bytes_transferred);
2015-06-02 17:03:22 +00:00
virtual void Write (const uint8_t * buf, size_t len); // can be overloaded
2014-08-13 19:25:52 +00:00
void HandleWrite (const boost::system::error_code& ecode);
void StreamReceive ();
void HandleStreamReceive (const boost::system::error_code& ecode, std::size_t bytes_transferred);
2014-08-20 19:03:10 +00:00
void HandleConnect (const boost::system::error_code& ecode);
2014-08-13 19:25:52 +00:00
private:
uint8_t m_Buffer[I2P_TUNNEL_CONNECTION_BUFFER_SIZE], m_StreamBuffer[I2P_TUNNEL_CONNECTION_BUFFER_SIZE];
std::shared_ptr<boost::asio::ip::tcp::socket> m_Socket;
2014-11-23 16:33:58 +00:00
std::shared_ptr<i2p::stream::Stream> m_Stream;
2014-11-24 03:23:17 +00:00
boost::asio::ip::tcp::endpoint m_RemoteEndpoint;
bool m_IsQuiet; // don't send destination
};
2015-06-02 17:03:22 +00:00
class I2PTunnelConnectionHTTP: public I2PTunnelConnection
{
public:
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);
protected:
void Write (const uint8_t * buf, size_t len);
private:
2015-06-03 16:30:15 +00:00
std::string m_Host;
std::stringstream m_InHeader, m_OutHeader;
bool m_HeaderSent;
2016-01-11 18:48:18 +00:00
std::shared_ptr<const i2p::data::IdentityEx> m_From;
2015-06-02 17:03:22 +00:00
};
2016-02-22 19:33:21 +00:00
class I2PTunnelConnectionIRC: public I2PTunnelConnection
{
public:
I2PTunnelConnectionIRC (I2PService * owner, std::shared_ptr<i2p::stream::Stream> stream,
std::shared_ptr<boost::asio::ip::tcp::socket> socket,
2016-03-04 06:37:38 +00:00
const boost::asio::ip::tcp::endpoint& target, const std::string& m_WebircPass);
2016-02-22 19:33:21 +00:00
protected:
void Write (const uint8_t * buf, size_t len);
private:
2016-03-05 01:35:53 +00:00
2016-02-22 19:33:21 +00:00
std::shared_ptr<const i2p::data::IdentityEx> m_From;
std::stringstream m_OutPacket, m_InPacket;
2016-03-05 01:35:53 +00:00
bool m_NeedsWebIrc;
std::string m_WebircPass;
2016-02-22 19:33:21 +00:00
};
2015-01-08 02:49:35 +00:00
class I2PClientTunnel: public TCPIPAcceptor
2014-08-13 01:14:19 +00:00
{
2015-01-08 02:49:35 +00:00
protected:
// Implements TCPIPAcceptor
std::shared_ptr<I2PServiceHandler> CreateHandler(std::shared_ptr<boost::asio::ip::tcp::socket> socket);
2015-01-08 02:49:35 +00:00
2014-08-13 01:14:19 +00:00
public:
2016-01-14 01:21:53 +00:00
I2PClientTunnel (const std::string& name, const std::string& destination,
const std::string& address, int port, std::shared_ptr<ClientDestination> localDestination, int destinationPort = 0);
2015-01-08 02:49:35 +00:00
~I2PClientTunnel () {}
2014-08-13 19:25:52 +00:00
void Start ();
void Stop ();
2016-01-14 01:21:53 +00:00
const char* GetName() { return m_Name.c_str (); }
2014-08-13 01:14:19 +00:00
private:
const i2p::data::IdentHash * GetIdentHash ();
2014-08-13 01:14:19 +00:00
2016-01-14 01:21:53 +00:00
private:
std::string m_Name, m_Destination;
2014-08-13 19:25:52 +00:00
const i2p::data::IdentHash * m_DestinationIdentHash;
2015-03-13 17:29:27 +00:00
int m_DestinationPort;
2016-08-21 19:02:17 +00:00
};
2014-08-20 19:03:10 +00:00
2016-08-21 19:02:17 +00:00
/** 2 minute timeout for udp sessions */
const uint64_t I2P_UDP_SESSION_TIMEOUT = 1000 * 60 * 2;
/** max size for i2p udp */
2016-08-22 21:19:22 +00:00
const size_t I2P_UDP_MAX_MTU = i2p::datagram::MAX_DATAGRAM_SIZE;
2016-08-21 19:02:17 +00:00
struct UDPSession
{
2016-08-22 17:54:00 +00:00
i2p::datagram::DatagramDestination * m_Destination;
boost::asio::io_service & m_Service;
2016-08-21 19:02:17 +00:00
boost::asio::ip::udp::socket IPSocket;
i2p::data::IdentHash Identity;
boost::asio::ip::udp::endpoint FromEndpoint;
2016-08-21 22:46:34 +00:00
boost::asio::ip::udp::endpoint SendEndpoint;
2016-08-21 19:02:17 +00:00
uint64_t LastActivity;
uint16_t LocalPort;
uint16_t RemotePort;
uint8_t m_Buffer[I2P_UDP_MAX_MTU];
2016-08-22 17:54:00 +00:00
UDPSession(boost::asio::ip::udp::endpoint localEndpoint, const std::shared_ptr<i2p::client::ClientDestination> & localDestination, boost::asio::ip::udp::endpoint remote, const i2p::data::IdentHash ident, uint16_t ourPort, uint16_t theirPort);
2016-08-21 19:02:17 +00:00
void HandleReceived(const boost::system::error_code & ecode, std::size_t len);
void Receive();
};
/** server side udp tunnel, many i2p inbound to 1 ip outbound */
class I2PUDPServerTunnel
{
public:
2016-08-22 17:54:00 +00:00
I2PUDPServerTunnel(const std::string & name, std::shared_ptr<i2p::client::ClientDestination> localDestination, boost::asio::ip::udp::endpoint forwardTo, uint16_t port);
2016-08-21 19:02:17 +00:00
~I2PUDPServerTunnel();
/** expire stale udp conversations */
void ExpireStale(const uint64_t delta=I2P_UDP_SESSION_TIMEOUT);
2016-08-22 17:04:36 +00:00
void Start();
2016-08-21 19:02:17 +00:00
private:
void HandleRecvFromI2P(const i2p::data::IdentityEx& from, uint16_t fromPort, uint16_t toPort, const uint8_t * buf, size_t len);
2016-08-22 17:54:00 +00:00
UDPSession * ObtainUDPSession(const i2p::data::IdentityEx& from, uint16_t localPort, uint16_t remotePort);
2016-08-21 19:02:17 +00:00
private:
const uint16_t LocalPort;
boost::asio::ip::udp::endpoint m_Endpoint;
std::mutex m_SessionsMutex;
2016-08-22 17:54:00 +00:00
std::vector<UDPSession*> m_Sessions;
2016-08-22 01:17:09 +00:00
std::shared_ptr<i2p::client::ClientDestination> m_LocalDest;
2016-08-21 19:02:17 +00:00
uint8_t m_Buffer[I2P_UDP_MAX_MTU];
};
class I2PUDPClientTunnel
{
public:
2016-08-22 17:54:00 +00:00
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);
2016-08-21 19:02:17 +00:00
~I2PUDPClientTunnel();
void Start();
private:
void HandleRecvFromI2P(const i2p::data::IdentityEx& from, uint16_t fromPort, uint16_t toPort, const uint8_t * buf, size_t len);
void TryResolving();
UDPSession * m_Session;
const std::string m_RemoteDest;
std::shared_ptr<i2p::client::ClientDestination> m_LocalDest;
const boost::asio::ip::udp::endpoint m_LocalEndpoint;
2016-08-21 22:46:34 +00:00
i2p::data::IdentHash * m_RemoteIdent;
2016-08-21 19:02:17 +00:00
std::thread * m_ResolveThread;
uint16_t LocalPort;
uint16_t RemotePort;
bool m_cancel_resolve;
};
class I2PServerTunnel: public I2PService
2014-08-20 19:03:10 +00:00
{
public:
2016-01-14 01:21:53 +00:00
I2PServerTunnel (const std::string& name, const std::string& address, int port,
std::shared_ptr<ClientDestination> localDestination, int inport = 0, bool gzip = true);
2014-08-20 19:03:10 +00:00
void Start ();
void Stop ();
2015-03-16 18:52:42 +00:00
void SetAccessList (const std::set<i2p::data::IdentHash>& accessList);
2015-06-02 17:03:22 +00:00
const std::string& GetAddress() const { return m_Address; }
int GetPort () const { return m_Port; };
uint16_t GetLocalPort () const { return m_PortDestination->GetLocalPort (); };
2015-06-02 17:03:22 +00:00
const boost::asio::ip::tcp::endpoint& GetEndpoint () const { return m_Endpoint; }
2016-01-14 01:21:53 +00:00
const char* GetName() { return m_Name.c_str (); }
2016-07-28 15:16:29 +00:00
void SetMaxConnsPerMinute(const uint32_t conns) { m_PortDestination->SetMaxConnsPerMinute(conns); }
2016-08-21 19:02:17 +00:00
2016-07-28 15:16:29 +00:00
private:
2014-08-20 19:03:10 +00:00
void HandleResolve (const boost::system::error_code& ecode, boost::asio::ip::tcp::resolver::iterator it,
std::shared_ptr<boost::asio::ip::tcp::resolver> resolver);
2014-08-20 19:03:10 +00:00
void Accept ();
2014-11-23 16:33:58 +00:00
void HandleAccept (std::shared_ptr<i2p::stream::Stream> stream);
2015-06-02 17:03:22 +00:00
virtual void CreateI2PConnection (std::shared_ptr<i2p::stream::Stream> stream);
2014-08-20 19:03:10 +00:00
private:
2016-01-14 01:21:53 +00:00
std::string m_Name, m_Address;
2015-06-02 17:03:22 +00:00
int m_Port;
boost::asio::ip::tcp::endpoint m_Endpoint;
2015-03-16 18:52:42 +00:00
std::shared_ptr<i2p::stream::StreamingDestination> m_PortDestination;
std::set<i2p::data::IdentHash> m_AccessList;
2016-08-21 19:02:17 +00:00
bool m_IsAccessList;
2014-08-20 19:03:10 +00:00
};
2015-05-20 20:00:09 +00:00
class I2PServerTunnelHTTP: public I2PServerTunnel
{
public:
2016-01-14 01:21:53 +00:00
I2PServerTunnelHTTP (const std::string& name, const std::string& address, int port,
2016-02-26 01:32:05 +00:00
std::shared_ptr<ClientDestination> localDestination, const std::string& host,
int inport = 0, bool gzip = true);
2015-06-02 17:03:22 +00:00
private:
2016-02-26 01:32:05 +00:00
void CreateI2PConnection (std::shared_ptr<i2p::stream::Stream> stream);
private:
std::string m_Host;
2015-05-20 20:00:09 +00:00
};
2016-02-22 19:33:21 +00:00
class I2PServerTunnelIRC: public I2PServerTunnel
{
public:
I2PServerTunnelIRC (const std::string& name, const std::string& address, int port,
2016-03-04 06:37:38 +00:00
std::shared_ptr<ClientDestination> localDestination, const std::string& webircpass,
int inport = 0, bool gzip = true);
private:
void CreateI2PConnection (std::shared_ptr<i2p::stream::Stream> stream);
2016-02-22 19:33:21 +00:00
private:
2016-03-04 06:37:38 +00:00
std::string m_WebircPass;
2016-02-22 19:33:21 +00:00
};
}
2014-08-13 01:14:19 +00:00
}
#endif