2020-05-22 13:18:41 +00:00
|
|
|
/*
|
2024-11-25 21:00:06 +00:00
|
|
|
* Copyright (c) 2013-2024, The PurpleI2P Project
|
2020-05-22 13:18:41 +00:00
|
|
|
*
|
|
|
|
* This file is part of Purple i2pd project and licensed under BSD3
|
|
|
|
*
|
|
|
|
* See full license text in LICENSE file at top of project tree
|
|
|
|
*/
|
|
|
|
|
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>
|
2016-11-17 15:36:27 +00:00
|
|
|
#include <tuple>
|
2014-11-23 16:33:58 +00:00
|
|
|
#include <memory>
|
2015-06-02 20:21:38 +00:00
|
|
|
#include <sstream>
|
2014-08-13 01:14:19 +00:00
|
|
|
#include <boost/asio.hpp>
|
2022-10-09 01:41:28 +00:00
|
|
|
#include <boost/asio/ssl.hpp>
|
2014-08-13 01:14:19 +00:00
|
|
|
#include "Identity.h"
|
2014-10-22 15:46:54 +00:00
|
|
|
#include "Destination.h"
|
2014-08-13 01:14:19 +00:00
|
|
|
#include "Streaming.h"
|
2015-01-07 18:09:59 +00:00
|
|
|
#include "I2PService.h"
|
2019-03-28 14:17:03 +00:00
|
|
|
#include "AddressBook.h"
|
2014-08-13 01:14:19 +00:00
|
|
|
|
|
|
|
namespace i2p
|
|
|
|
{
|
2014-10-16 14:28:44 +00:00
|
|
|
namespace client
|
2014-08-13 01:14:19 +00:00
|
|
|
{
|
2016-12-23 12:27:34 +00:00
|
|
|
const size_t I2P_TUNNEL_CONNECTION_BUFFER_SIZE = 65536;
|
2017-04-08 16:51:35 +00:00
|
|
|
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
|
2017-04-08 16:51:35 +00:00
|
|
|
// for HTTP tunnels
|
2022-05-20 16:56:05 +00:00
|
|
|
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
|
2022-08-23 23:06:28 +00:00
|
|
|
const int I2P_TUNNEL_HTTP_MAX_HEADER_SIZE = 8192;
|
2022-10-09 17:24:43 +00:00
|
|
|
|
2015-01-07 18:09:59 +00:00
|
|
|
class I2PTunnelConnection: public I2PServiceHandler, public std::enable_shared_from_this<I2PTunnelConnection>
|
2014-08-13 01:14:19 +00:00
|
|
|
{
|
|
|
|
public:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2015-04-06 18:41:07 +00:00
|
|
|
I2PTunnelConnection (I2PService * owner, std::shared_ptr<boost::asio::ip::tcp::socket> socket,
|
2023-06-12 02:10:32 +00:00
|
|
|
std::shared_ptr<const i2p::data::LeaseSet> leaseSet, uint16_t port = 0); // to I2P
|
2015-04-06 18:41:07 +00:00
|
|
|
I2PTunnelConnection (I2PService * owner, std::shared_ptr<boost::asio::ip::tcp::socket> socket,
|
2017-04-08 16:51:35 +00:00
|
|
|
std::shared_ptr<i2p::stream::Stream> stream); // to I2P using simplified API
|
2022-10-09 01:41:28 +00:00
|
|
|
I2PTunnelConnection (I2PService * owner, std::shared_ptr<i2p::stream::Stream> stream,
|
|
|
|
const boost::asio::ip::tcp::endpoint& target, bool quiet = true,
|
|
|
|
std::shared_ptr<boost::asio::ssl::context> sslCtx = nullptr); // from I2P
|
2014-08-13 01:14:19 +00:00
|
|
|
~I2PTunnelConnection ();
|
2014-12-04 01:37:20 +00:00
|
|
|
void I2PConnect (const uint8_t * msg = nullptr, size_t len = 0);
|
2017-01-12 21:17:11 +00:00
|
|
|
void Connect (bool isUniqueLocal = true);
|
2021-02-19 20:15:58 +00:00
|
|
|
void Connect (const boost::asio::ip::address& localAddress);
|
2021-11-27 20:30:35 +00:00
|
|
|
|
2015-06-02 17:03:22 +00:00
|
|
|
protected:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2017-04-08 16:51:35 +00:00
|
|
|
void Terminate ();
|
2014-08-13 19:25:52 +00:00
|
|
|
|
|
|
|
void Receive ();
|
2022-10-09 01:41:28 +00:00
|
|
|
void StreamReceive ();
|
2015-06-02 17:03:22 +00:00
|
|
|
virtual void Write (const uint8_t * buf, size_t len); // can be overloaded
|
2020-10-06 20:22:40 +00:00
|
|
|
virtual void WriteToStream (const uint8_t * buf, size_t len); // can be overloaded
|
2021-11-27 20:30:35 +00:00
|
|
|
|
2022-10-09 01:41:28 +00:00
|
|
|
std::shared_ptr<boost::asio::ip::tcp::socket> GetSocket () const { return m_Socket; };
|
|
|
|
std::shared_ptr<boost::asio::ssl::stream<boost::asio::ip::tcp::socket&> > GetSSL () const { return m_SSL; };
|
2014-08-13 19:25:52 +00:00
|
|
|
|
2022-10-09 01:41:28 +00:00
|
|
|
private:
|
2017-01-20 15:02:16 +00:00
|
|
|
|
2022-10-09 01:41:28 +00:00
|
|
|
void HandleConnect (const boost::system::error_code& ecode);
|
|
|
|
void HandleHandshake (const boost::system::error_code& ecode);
|
|
|
|
void Established ();
|
|
|
|
void HandleReceive (const boost::system::error_code& ecode, std::size_t bytes_transferred);
|
|
|
|
void HandleWrite (const boost::system::error_code& ecode);
|
|
|
|
void HandleStreamReceive (const boost::system::error_code& ecode, std::size_t bytes_transferred);
|
2022-10-09 17:24:43 +00:00
|
|
|
|
2014-08-13 19:25:52 +00:00
|
|
|
private:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2014-08-13 19:25:52 +00:00
|
|
|
uint8_t m_Buffer[I2P_TUNNEL_CONNECTION_BUFFER_SIZE], m_StreamBuffer[I2P_TUNNEL_CONNECTION_BUFFER_SIZE];
|
2015-04-06 18:41:07 +00:00
|
|
|
std::shared_ptr<boost::asio::ip::tcp::socket> m_Socket;
|
2022-10-09 01:41:28 +00:00
|
|
|
std::shared_ptr<boost::asio::ssl::stream<boost::asio::ip::tcp::socket&> > m_SSL;
|
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;
|
2014-12-05 19:46:59 +00:00
|
|
|
bool m_IsQuiet; // don't send destination
|
2015-01-07 18:09:59 +00:00
|
|
|
};
|
2014-09-10 02:40:12 +00:00
|
|
|
|
2017-02-07 02:39:15 +00:00
|
|
|
class I2PClientTunnelConnectionHTTP: public I2PTunnelConnection
|
|
|
|
{
|
|
|
|
public:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2017-02-07 02:39:15 +00:00
|
|
|
I2PClientTunnelConnectionHTTP (I2PService * owner, std::shared_ptr<boost::asio::ip::tcp::socket> socket,
|
|
|
|
std::shared_ptr<i2p::stream::Stream> stream):
|
|
|
|
I2PTunnelConnection (owner, socket, stream), m_HeaderSent (false),
|
|
|
|
m_ConnectionSent (false), m_ProxyConnectionSent (false) {};
|
|
|
|
|
|
|
|
protected:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2017-04-08 16:51:35 +00:00
|
|
|
void Write (const uint8_t * buf, size_t len);
|
|
|
|
|
|
|
|
private:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2017-02-07 02:39:15 +00:00
|
|
|
std::stringstream m_InHeader, m_OutHeader;
|
|
|
|
bool m_HeaderSent, m_ConnectionSent, m_ProxyConnectionSent;
|
2017-04-08 16:51:35 +00:00
|
|
|
};
|
|
|
|
|
2017-02-07 02:39:15 +00:00
|
|
|
class I2PServerTunnelConnectionHTTP: public I2PTunnelConnection
|
2015-06-02 17:03:22 +00:00
|
|
|
{
|
|
|
|
public:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2017-02-07 02:39:15 +00:00
|
|
|
I2PServerTunnelConnectionHTTP (I2PService * owner, std::shared_ptr<i2p::stream::Stream> stream,
|
2022-10-09 01:41:28 +00:00
|
|
|
const boost::asio::ip::tcp::endpoint& target, const std::string& host,
|
|
|
|
std::shared_ptr<boost::asio::ssl::context> sslCtx = nullptr);
|
2015-06-02 20:21:38 +00:00
|
|
|
|
|
|
|
protected:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2015-06-02 20:21:38 +00:00
|
|
|
void Write (const uint8_t * buf, size_t len);
|
2021-11-27 20:30:35 +00:00
|
|
|
void WriteToStream (const uint8_t * buf, size_t len);
|
2015-06-02 20:21:38 +00:00
|
|
|
|
|
|
|
private:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2015-06-03 16:30:15 +00:00
|
|
|
std::string m_Host;
|
2022-05-29 20:59:15 +00:00
|
|
|
std::stringstream m_InHeader, m_OutHeader;
|
2020-10-06 20:22:40 +00:00
|
|
|
bool m_HeaderSent, m_ResponseHeaderSent;
|
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
|
2017-10-04 17:15:29 +00:00
|
|
|
{
|
|
|
|
public:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2017-10-04 17:15:29 +00:00
|
|
|
I2PTunnelConnectionIRC (I2PService * owner, std::shared_ptr<i2p::stream::Stream> stream,
|
2022-10-10 15:02:19 +00:00
|
|
|
const boost::asio::ip::tcp::endpoint& target, const std::string& m_WebircPass,
|
|
|
|
std::shared_ptr<boost::asio::ssl::context> sslCtx = nullptr);
|
2016-02-22 19:33:21 +00:00
|
|
|
|
2017-10-04 17:15:29 +00:00
|
|
|
protected:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2017-10-04 17:15:29 +00:00
|
|
|
void Write (const uint8_t * buf, size_t len);
|
2017-04-08 16:51:35 +00:00
|
|
|
|
2017-10-04 17:15:29 +00:00
|
|
|
private:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2017-10-04 17:15:29 +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;
|
2017-10-04 17:15:29 +00:00
|
|
|
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:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2015-01-08 02:49:35 +00:00
|
|
|
// Implements TCPIPAcceptor
|
2015-04-06 18:41:07 +00:00
|
|
|
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:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2017-04-08 16:51:35 +00:00
|
|
|
I2PClientTunnel (const std::string& name, const std::string& destination,
|
2023-06-12 02:10:32 +00:00
|
|
|
const std::string& address, uint16_t port, std::shared_ptr<ClientDestination> localDestination, uint16_t destinationPort = 0);
|
2015-01-08 02:49:35 +00:00
|
|
|
~I2PClientTunnel () {}
|
2017-04-08 16:51:35 +00:00
|
|
|
|
2014-08-13 19:25:52 +00:00
|
|
|
void Start ();
|
|
|
|
void Stop ();
|
|
|
|
|
2017-04-08 16:51:35 +00:00
|
|
|
const char* GetName() { return m_Name.c_str (); }
|
2021-09-26 20:25:12 +00:00
|
|
|
void SetKeepAliveInterval (uint32_t keepAliveInterval);
|
2021-11-27 20:30:35 +00:00
|
|
|
|
2014-08-13 01:14:19 +00:00
|
|
|
private:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2019-03-28 14:17:03 +00:00
|
|
|
std::shared_ptr<const Address> GetAddress ();
|
2021-11-27 20:30:35 +00:00
|
|
|
|
2021-09-26 20:25:12 +00:00
|
|
|
void ScheduleKeepAliveTimer ();
|
|
|
|
void HandleKeepAliveTimer (const boost::system::error_code& ecode);
|
2014-08-13 01:14:19 +00:00
|
|
|
|
2016-01-14 01:21:53 +00:00
|
|
|
private:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2016-01-14 01:21:53 +00:00
|
|
|
std::string m_Name, m_Destination;
|
2019-03-28 14:17:03 +00:00
|
|
|
std::shared_ptr<const Address> m_Address;
|
2023-06-12 02:10:32 +00:00
|
|
|
uint16_t m_DestinationPort;
|
2021-09-26 20:25:12 +00:00
|
|
|
uint32_t m_KeepAliveInterval;
|
|
|
|
std::unique_ptr<boost::asio::deadline_timer> m_KeepAliveTimer;
|
2016-08-21 19:02:17 +00:00
|
|
|
};
|
|
|
|
|
2015-01-07 18:09:59 +00:00
|
|
|
class I2PServerTunnel: public I2PService
|
2014-08-20 19:03:10 +00:00
|
|
|
{
|
|
|
|
public:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2023-06-12 02:10:32 +00:00
|
|
|
I2PServerTunnel (const std::string& name, const std::string& address, uint16_t port,
|
|
|
|
std::shared_ptr<ClientDestination> localDestination, uint16_t inport = 0, bool gzip = true);
|
2014-08-20 19:03:10 +00:00
|
|
|
|
|
|
|
void Start ();
|
|
|
|
void Stop ();
|
|
|
|
|
2017-04-08 16:51:35 +00:00
|
|
|
void SetAccessList (const std::set<i2p::data::IdentHash>& accessList);
|
2015-03-16 18:52:42 +00:00
|
|
|
|
2017-01-12 21:17:11 +00:00
|
|
|
void SetUniqueLocal (bool isUniqueLocal) { m_IsUniqueLocal = isUniqueLocal; }
|
2017-01-13 18:47:51 +00:00
|
|
|
bool IsUniqueLocal () const { return m_IsUniqueLocal; }
|
2016-12-25 13:56:47 +00:00
|
|
|
|
2022-10-09 17:24:43 +00:00
|
|
|
void SetSSL (bool ssl);
|
2022-10-09 01:41:28 +00:00
|
|
|
std::shared_ptr<boost::asio::ssl::context> GetSSLCtx () const { return m_SSLCtx; };
|
2022-10-09 17:24:43 +00:00
|
|
|
|
2021-02-19 20:15:58 +00:00
|
|
|
void SetLocalAddress (const std::string& localAddress);
|
2021-11-27 20:30:35 +00:00
|
|
|
|
2015-06-02 17:03:22 +00:00
|
|
|
const std::string& GetAddress() const { return m_Address; }
|
2023-06-12 02:10:32 +00:00
|
|
|
uint16_t GetPort () const { return m_Port; };
|
2016-01-17 23:55:09 +00:00
|
|
|
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; }
|
|
|
|
|
2017-04-08 16:51:35 +00:00
|
|
|
const char* GetName() { return m_Name.c_str (); }
|
2016-07-28 15:16:29 +00:00
|
|
|
|
2017-10-04 17:15:29 +00:00
|
|
|
private:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2024-11-25 21:00:06 +00:00
|
|
|
void HandleResolve (const boost::system::error_code& ecode, boost::asio::ip::tcp::resolver::results_type endpoints,
|
2015-06-02 17:18:41 +00:00
|
|
|
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);
|
2017-01-13 18:47:51 +00:00
|
|
|
virtual std::shared_ptr<I2PTunnelConnection> CreateI2PConnection (std::shared_ptr<i2p::stream::Stream> stream);
|
2014-08-20 19:03:10 +00:00
|
|
|
|
|
|
|
private:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2017-01-12 21:17:11 +00:00
|
|
|
bool m_IsUniqueLocal;
|
2016-01-14 01:21:53 +00:00
|
|
|
std::string m_Name, m_Address;
|
2023-06-12 02:10:32 +00:00
|
|
|
uint16_t m_Port;
|
2017-04-08 16:51:35 +00:00
|
|
|
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;
|
2021-02-19 20:15:58 +00:00
|
|
|
std::unique_ptr<boost::asio::ip::address> m_LocalAddress;
|
2022-10-09 01:41:28 +00:00
|
|
|
std::shared_ptr<boost::asio::ssl::context> m_SSLCtx;
|
2014-08-20 19:03:10 +00:00
|
|
|
};
|
2015-05-20 20:00:09 +00:00
|
|
|
|
|
|
|
class I2PServerTunnelHTTP: public I2PServerTunnel
|
|
|
|
{
|
|
|
|
public:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2023-06-12 02:10:32 +00:00
|
|
|
I2PServerTunnelHTTP (const std::string& name, const std::string& address, uint16_t port,
|
2016-02-26 01:32:05 +00:00
|
|
|
std::shared_ptr<ClientDestination> localDestination, const std::string& host,
|
2023-06-12 02:10:32 +00:00
|
|
|
uint16_t inport = 0, bool gzip = true);
|
2015-06-02 17:03:22 +00:00
|
|
|
|
|
|
|
private:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2017-01-13 18:47:51 +00:00
|
|
|
std::shared_ptr<I2PTunnelConnection> CreateI2PConnection (std::shared_ptr<i2p::stream::Stream> stream);
|
2016-02-26 01:32:05 +00:00
|
|
|
|
|
|
|
private:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2016-02-26 01:32:05 +00:00
|
|
|
std::string m_Host;
|
2015-05-20 20:00:09 +00:00
|
|
|
};
|
2017-04-08 16:51:35 +00:00
|
|
|
|
2017-10-04 17:15:29 +00:00
|
|
|
class I2PServerTunnelIRC: public I2PServerTunnel
|
|
|
|
{
|
|
|
|
public:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2023-06-12 02:10:32 +00:00
|
|
|
I2PServerTunnelIRC (const std::string& name, const std::string& address, uint16_t port,
|
2017-10-04 17:15:29 +00:00
|
|
|
std::shared_ptr<ClientDestination> localDestination, const std::string& webircpass,
|
2023-06-12 02:10:32 +00:00
|
|
|
uint16_t inport = 0, bool gzip = true);
|
2016-02-22 19:33:21 +00:00
|
|
|
|
2017-10-04 17:15:29 +00:00
|
|
|
private:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2017-10-04 17:15:29 +00:00
|
|
|
std::shared_ptr<I2PTunnelConnection> CreateI2PConnection (std::shared_ptr<i2p::stream::Stream> stream);
|
2016-02-22 19:33:21 +00:00
|
|
|
|
2017-10-04 17:15:29 +00:00
|
|
|
private:
|
2020-03-01 10:25:50 +00:00
|
|
|
|
2017-10-04 17:15:29 +00:00
|
|
|
std::string m_WebircPass;
|
|
|
|
};
|
2022-10-25 19:30:12 +00:00
|
|
|
|
|
|
|
boost::asio::ip::address GetLoopbackAddressFor(const i2p::data::IdentHash & addr);
|
2015-01-07 18:09:59 +00:00
|
|
|
}
|
2017-04-08 16:51:35 +00:00
|
|
|
}
|
2014-08-13 01:14:19 +00:00
|
|
|
|
|
|
|
#endif
|