2019-05-16 08:17:05 +03:00
/*
2025-01-12 18:36:35 -05:00
* Copyright ( c ) 2013 - 2025 , The PurpleI2P Project
2019-05-16 08:17:05 +03: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
*/
2015-01-07 00:15:38 +01:00
# include <cstring>
# include <cassert>
2015-01-07 19:09:59 +01:00
# include <string>
2024-09-02 21:05:40 -04:00
# include <string_view>
2015-01-07 19:09:59 +01:00
# include <atomic>
2016-06-23 00:00:00 +00:00
# include <memory>
# include <set>
# include <boost/asio.hpp>
2019-05-16 08:17:05 +03:00
# include <boost/algorithm/string/predicate.hpp>
2016-06-23 00:00:00 +00:00
# include <mutex>
# include "I2PService.h"
# include "Destination.h"
2014-03-17 23:31:29 +01:00
# include "HTTPProxy.h"
2015-02-07 18:34:25 +01:00
# include "util.h"
2015-01-07 00:15:38 +01:00
# include "Identity.h"
2015-01-07 19:09:59 +01:00
# include "Streaming.h"
2015-01-07 00:15:38 +01:00
# include "Destination.h"
# include "ClientContext.h"
# include "I2PEndian.h"
2015-01-07 19:09:59 +01:00
# include "I2PTunnel.h"
2016-03-11 15:30:50 +05:00
# include "Config.h"
2016-05-27 00:00:00 +00:00
# include "HTTP.h"
2021-05-23 06:06:04 +03:00
# include "I18N.h"
2024-02-11 11:23:37 -05:00
# include "Socks5.h"
2014-03-17 23:31:29 +01:00
2016-06-23 00:00:00 +00:00
namespace i2p {
namespace proxy {
2022-05-20 20:49:26 +03:00
static const std : : vector < std : : string > jumporder = {
" reg.i2p " ,
" stats.i2p " ,
" identiguy.i2p " ,
2022-11-14 15:29:16 -05:00
" notbob.i2p "
2022-05-20 20:49:26 +03:00
} ;
static const std : : map < std : : string , std : : string > jumpservices = {
2021-01-27 06:48:35 +03:00
{ " reg.i2p " , " http://shx5vqsw7usdaunyzr2qmes2fq37oumybpudrd4jjj4e4vk4uusa.b32.i2p/jump/ " } ,
{ " identiguy.i2p " , " http://3mzmrus2oron5fxptw7hw2puho3bnqmw2hqy7nw64dsrrjwdilva.b32.i2p/cgi-bin/query?hostname= " } ,
{ " stats.i2p " , " http://7tbay5p4kzeekxvyvbf6v7eauazemsnnl2aoyqhg5jzpr5eke7tq.b32.i2p/cgi-bin/jump.cgi?a= " } ,
2022-11-14 15:29:16 -05:00
{ " notbob.i2p " , " http://nytzrhrjjfsutowojvxi7hphesskpqqr65wpistz6wa7cpajhp7a.b32.i2p/cgi-bin/jump.cgi?q= " }
2016-07-14 00:00:00 +00:00
} ;
static const char * pageHead =
" <head> \r \n "
2021-02-09 08:32:35 +03:00
" <meta name= \" viewport \" content= \" width=device-width, initial-scale=1.0 \" > \r \n "
2016-11-17 06:04:29 +03:00
" <title>I2Pd HTTP proxy</title> \r \n "
2016-07-14 00:00:00 +00:00
" <style type= \" text/css \" > \r \n "
" body { font: 100%/1.5em sans-serif; margin: 0; padding: 1.5em; background: #FAFAFA; color: #103456; } \r \n "
2021-02-09 08:32:35 +03:00
" h1 { font-size: 1.7em; color: #894C84; } \r \n "
" @media screen and (max-width: 980px) { h1 { font-size: 1.7em; text-align: center; color: #894C84; }} \r \n "
2016-07-14 00:00:00 +00:00
" </style> \r \n "
" </head> \r \n "
;
2025-01-12 18:36:35 -05:00
static bool str_rmatch ( std : : string_view str , const char * suffix )
2024-09-02 21:05:40 -04:00
{
2016-06-27 01:30:00 +00:00
auto pos = str . rfind ( suffix ) ;
if ( pos = = std : : string : : npos )
return false ; /* not found */
if ( str . length ( ) = = ( pos + std : : strlen ( suffix ) ) )
return true ; /* match */
return false ;
}
2016-06-23 00:00:00 +00:00
class HTTPReqHandler : public i2p : : client : : I2PServiceHandler , public std : : enable_shared_from_this < HTTPReqHandler >
2015-03-17 11:44:01 -04:00
{
2015-01-07 19:09:59 +01:00
private :
2016-07-01 00:00:00 +00:00
bool HandleRequest ( ) ;
2015-01-07 19:09:59 +01:00
void HandleSockRecv ( const boost : : system : : error_code & ecode , std : : size_t bytes_transfered ) ;
void Terminate ( ) ;
void AsyncSockRead ( ) ;
2023-10-29 22:11:38 -04:00
static bool ExtractAddressHelper ( i2p : : http : : URL & url , std : : string & jump , bool & confirm ) ;
2024-09-02 21:05:40 -04:00
static bool VerifyAddressHelper ( std : : string_view jump ) ;
2024-09-22 21:07:44 -04:00
void SanitizeHTTPRequest ( i2p : : http : : HTTPReq & req ) ;
2015-01-07 19:09:59 +01:00
void SentHTTPFailed ( const boost : : system : : error_code & ecode ) ;
void HandleStreamRequestComplete ( std : : shared_ptr < i2p : : stream : : Stream > stream ) ;
2016-07-14 00:00:00 +00:00
/* error helpers */
2025-01-12 18:36:35 -05:00
void GenericProxyError ( std : : string_view title , std : : string_view description ) ;
void GenericProxyInfo ( std : : string_view title , std : : string_view description ) ;
void HostNotFound ( std : : string_view host ) ;
void SendProxyError ( std : : string_view content ) ;
2024-09-02 21:05:40 -04:00
void SendRedirect ( const std : : string & address ) ;
2015-01-07 19:09:59 +01:00
2019-04-25 23:06:14 +03:00
void ForwardToUpstreamProxy ( ) ;
void HandleUpstreamHTTPProxyConnect ( const boost : : system : : error_code & ec ) ;
void HandleUpstreamSocksProxyConnect ( const boost : : system : : error_code & ec ) ;
2025-01-12 18:36:35 -05:00
void HTTPConnect ( std : : string_view host , uint16_t port ) ;
2019-04-25 23:06:14 +03:00
void HandleHTTPConnectStreamRequestComplete ( std : : shared_ptr < i2p : : stream : : Stream > stream ) ;
2017-09-03 09:46:55 -04:00
2019-04-25 23:06:14 +03:00
typedef std : : function < void ( boost : : asio : : ip : : tcp : : endpoint ) > ProxyResolvedHandler ;
2017-09-03 09:46:55 -04:00
2024-11-25 16:00:06 -05:00
void HandleUpstreamProxyResolved ( const boost : : system : : error_code & ecode , boost : : asio : : ip : : tcp : : resolver : : results_type endpoints , ProxyResolvedHandler handler ) ;
2016-11-20 09:25:56 -05:00
2019-04-25 23:06:14 +03:00
void SocksProxySuccess ( ) ;
void HandoverToUpstreamProxy ( ) ;
2017-09-03 09:46:55 -04:00
2016-07-01 00:00:00 +00:00
uint8_t m_recv_chunk [ 8192 ] ;
std : : string m_recv_buf ; // from client
std : : string m_send_buf ; // to upstream
2015-04-06 14:41:07 -04:00
std : : shared_ptr < boost : : asio : : ip : : tcp : : socket > m_sock ;
2019-04-25 23:06:14 +03:00
std : : shared_ptr < boost : : asio : : ip : : tcp : : socket > m_proxysock ;
boost : : asio : : ip : : tcp : : resolver m_proxy_resolver ;
2023-10-30 10:09:47 -04:00
std : : string m_OutproxyUrl , m_Response ;
2024-09-22 21:07:44 -04:00
bool m_Addresshelper , m_SendUserAgent ;
2019-04-25 23:06:14 +03:00
i2p : : http : : URL m_ProxyURL ;
i2p : : http : : URL m_RequestURL ;
2023-02-28 19:41:17 +02:00
int m_req_len ;
2019-04-25 23:06:14 +03:00
i2p : : http : : URL m_ClientRequestURL ;
i2p : : http : : HTTPReq m_ClientRequest ;
i2p : : http : : HTTPRes m_ClientResponse ;
std : : stringstream m_ClientRequestBuffer ;
2015-01-07 19:09:59 +01:00
public :
2015-03-17 11:44:01 -04:00
2016-06-23 00:00:00 +00:00
HTTPReqHandler ( HTTPProxy * parent , std : : shared_ptr < boost : : asio : : ip : : tcp : : socket > sock ) :
2016-11-20 09:25:56 -05:00
I2PServiceHandler ( parent ) , m_sock ( sock ) ,
m_proxysock ( std : : make_shared < boost : : asio : : ip : : tcp : : socket > ( parent - > GetService ( ) ) ) ,
2017-10-27 08:42:54 -04:00
m_proxy_resolver ( parent - > GetService ( ) ) ,
2019-04-25 23:06:14 +03:00
m_OutproxyUrl ( parent - > GetOutproxyURL ( ) ) ,
2024-09-22 21:07:44 -04:00
m_Addresshelper ( parent - > GetHelperSupport ( ) ) ,
m_SendUserAgent ( parent - > GetSendUserAgent ( ) ) { }
2016-06-23 00:00:00 +00:00
~ HTTPReqHandler ( ) { Terminate ( ) ; }
2016-07-01 00:00:00 +00:00
void Handle ( ) { AsyncSockRead ( ) ; } /* overload */
2015-01-07 19:09:59 +01:00
} ;
2016-06-23 00:00:00 +00:00
void HTTPReqHandler : : AsyncSockRead ( )
2015-01-07 00:15:38 +01:00
{
2021-11-27 22:53:53 +03:00
LogPrint ( eLogDebug , " HTTPProxy: Async sock read " ) ;
2016-06-27 13:00:00 +00:00
if ( ! m_sock ) {
2021-11-27 22:53:53 +03:00
LogPrint ( eLogError , " HTTPProxy: No socket for read " ) ;
2016-06-27 13:00:00 +00:00
return ;
2015-01-07 00:15:38 +01:00
}
2016-07-01 00:00:00 +00:00
m_sock - > async_read_some ( boost : : asio : : buffer ( m_recv_chunk , sizeof ( m_recv_chunk ) ) ,
2019-04-25 23:06:14 +03:00
std : : bind ( & HTTPReqHandler : : HandleSockRecv , shared_from_this ( ) ,
std : : placeholders : : _1 , std : : placeholders : : _2 ) ) ;
2015-01-07 00:15:38 +01:00
}
2016-06-23 00:00:00 +00:00
void HTTPReqHandler : : Terminate ( ) {
2015-01-07 19:09:59 +01:00
if ( Kill ( ) ) return ;
2017-09-03 09:46:55 -04:00
if ( m_sock )
2015-04-06 14:41:07 -04:00
{
2021-11-27 22:53:53 +03:00
LogPrint ( eLogDebug , " HTTPProxy: Close sock " ) ;
2015-01-07 00:15:38 +01:00
m_sock - > close ( ) ;
m_sock = nullptr ;
2014-03-17 23:31:29 +01:00
}
2016-11-20 09:25:56 -05:00
if ( m_proxysock )
{
2021-11-27 22:53:53 +03:00
LogPrint ( eLogDebug , " HTTPProxy: Close proxysock " ) ;
2016-11-20 09:25:56 -05:00
if ( m_proxysock - > is_open ( ) )
m_proxysock - > close ( ) ;
m_proxysock = nullptr ;
}
2015-01-07 19:09:59 +01:00
Done ( shared_from_this ( ) ) ;
2015-01-07 00:15:38 +01:00
}
2025-01-12 18:36:35 -05:00
void HTTPReqHandler : : GenericProxyError ( std : : string_view title , std : : string_view description )
{
2016-07-14 00:00:00 +00:00
std : : stringstream ss ;
2021-05-23 06:06:04 +03:00
ss < < " <h1> " < < tr ( " Proxy error " ) < < " : " < < title < < " </h1> \r \n " ;
2016-07-14 00:00:00 +00:00
ss < < " <p> " < < description < < " </p> \r \n " ;
2025-01-12 18:36:35 -05:00
SendProxyError ( ss . str ( ) ) ;
2016-08-05 21:23:54 +03:00
}
2016-07-14 00:00:00 +00:00
2025-01-12 18:36:35 -05:00
void HTTPReqHandler : : GenericProxyInfo ( std : : string_view title , std : : string_view description )
{
2016-11-17 06:04:29 +03:00
std : : stringstream ss ;
2021-05-23 06:06:04 +03:00
ss < < " <h1> " < < tr ( " Proxy info " ) < < " : " < < title < < " </h1> \r \n " ;
2016-11-17 06:04:29 +03:00
ss < < " <p> " < < description < < " </p> \r \n " ;
2025-01-12 18:36:35 -05:00
SendProxyError ( ss . str ( ) ) ;
2016-11-17 06:04:29 +03:00
}
2025-01-12 18:36:35 -05:00
void HTTPReqHandler : : HostNotFound ( std : : string_view host )
2024-09-02 21:05:40 -04:00
{
2016-07-14 00:00:00 +00:00
std : : stringstream ss ;
2021-05-23 06:06:04 +03:00
ss < < " <h1> " < < tr ( " Proxy error: Host not found " ) < < " </h1> \r \n "
< < " <p> " < < tr ( " Remote host not found in router's addressbook " ) < < " </p> \r \n "
< < " <p> " < < tr ( " You may try to find this host on jump services below " ) < < " :</p> \r \n "
2016-07-14 00:00:00 +00:00
< < " <ul> \r \n " ;
2022-05-20 21:04:41 +03:00
for ( const auto & jump : jumporder )
{
auto js = jumpservices . find ( jump ) ;
if ( js ! = jumpservices . end ( ) )
2022-05-24 07:09:24 -04:00
ss < < " <li><a href= \" " < < js - > second < < host < < " \" > " < < js - > first < < " </a></li> \r \n " ;
2016-07-14 00:00:00 +00:00
}
ss < < " </ul> \r \n " ;
2025-01-12 18:36:35 -05:00
SendProxyError ( ss . str ( ) ) ;
2016-08-05 21:23:54 +03:00
}
2015-01-07 00:15:38 +01:00
2025-01-12 18:36:35 -05:00
void HTTPReqHandler : : SendProxyError ( std : : string_view content )
2016-03-11 15:30:50 +05:00
{
2016-06-27 01:30:00 +00:00
i2p : : http : : HTTPRes res ;
2016-07-14 00:00:00 +00:00
res . code = 500 ;
res . add_header ( " Content-Type " , " text/html; charset=UTF-8 " ) ;
2016-07-01 00:00:00 +00:00
res . add_header ( " Connection " , " close " ) ;
2016-07-14 00:00:00 +00:00
std : : stringstream ss ;
ss < < " <html> \r \n " < < pageHead
< < " <body> " < < content < < " </body> \r \n "
< < " </html> \r \n " ;
res . body = ss . str ( ) ;
2023-10-30 10:09:47 -04:00
m_Response = res . to_string ( ) ;
boost : : asio : : async_write ( * m_sock , boost : : asio : : buffer ( m_Response ) , boost : : asio : : transfer_all ( ) ,
2020-03-01 13:25:50 +03:00
std : : bind ( & HTTPReqHandler : : SentHTTPFailed , shared_from_this ( ) , std : : placeholders : : _1 ) ) ;
2016-06-10 14:01:39 -04:00
}
2016-05-27 00:00:00 +00:00
2024-09-02 21:05:40 -04:00
void HTTPReqHandler : : SendRedirect ( const std : : string & address )
2022-10-09 22:13:49 +03:00
{
i2p : : http : : HTTPRes res ;
res . code = 302 ;
res . add_header ( " Location " , address ) ;
res . add_header ( " Connection " , " close " ) ;
2023-10-30 10:09:47 -04:00
m_Response = res . to_string ( ) ;
boost : : asio : : async_write ( * m_sock , boost : : asio : : buffer ( m_Response ) , boost : : asio : : transfer_all ( ) ,
2022-10-09 22:13:49 +03:00
std : : bind ( & HTTPReqHandler : : SentHTTPFailed , shared_from_this ( ) , std : : placeholders : : _1 ) ) ;
}
2023-10-29 22:11:38 -04:00
bool HTTPReqHandler : : ExtractAddressHelper ( i2p : : http : : URL & url , std : : string & jump , bool & confirm )
2015-03-17 11:44:01 -04:00
{
2017-11-16 01:13:42 +03:00
confirm = false ;
2016-07-01 00:00:00 +00:00
const char * param = " i2paddresshelper= " ;
std : : size_t pos = url . query . find ( param ) ;
std : : size_t len = std : : strlen ( param ) ;
std : : map < std : : string , std : : string > params ;
2017-11-16 01:13:42 +03:00
2016-07-01 00:00:00 +00:00
if ( pos = = std : : string : : npos )
return false ; /* not found */
if ( ! url . parse_query ( params ) )
return false ;
std : : string value = params [ " i2paddresshelper " ] ;
len + = value . length ( ) ;
2023-10-29 22:11:38 -04:00
jump = i2p : : http : : UrlDecode ( value ) ;
if ( ! VerifyAddressHelper ( jump ) )
{
LogPrint ( eLogError , " HTTPProxy: Malformed jump link " , jump ) ;
return false ;
}
2017-11-16 01:13:42 +03:00
// if we need update exists, request formed with update param
2023-02-14 18:51:14 +00:00
if ( params [ " update " ] = = " true " )
{
len + = std : : strlen ( " &update=true " ) ;
confirm = true ;
}
// if helper is not only one query option and it placed after user's query
if ( pos ! = 0 & & url . query [ pos - 1 ] = = ' & ' )
{
pos - - ;
len + + ;
}
// if helper is not only one query option and it placed before user's query
else if ( pos = = 0 & & url . query . length ( ) > len & & url . query [ len ] = = ' & ' )
{
// we don't touch the '?' but remove the trailing '&'
len + + ;
}
else
{
// there is no more query options, resetting hasquery flag
url . hasquery = false ;
}
// reset hasquery flag and remove addresshelper from URL
2021-05-23 13:16:52 +03:00
url . query . replace ( pos , len , " " ) ;
2016-07-01 00:00:00 +00:00
return true ;
2016-05-27 00:00:00 +00:00
}
2024-09-02 21:05:40 -04:00
bool HTTPReqHandler : : VerifyAddressHelper ( std : : string_view jump )
2023-10-29 22:11:38 -04:00
{
auto pos = jump . find ( " .b32.i2p " ) ;
if ( pos ! = std : : string : : npos )
{
auto b32 = jump . substr ( 0 , pos ) ;
for ( auto & ch : b32 )
if ( ! i2p : : data : : IsBase32 ( ch ) ) return false ;
return true ;
}
else
{
2023-10-31 09:10:56 -04:00
bool padding = false ;
2023-10-29 22:11:38 -04:00
for ( auto & ch : jump )
2023-10-31 09:10:56 -04:00
{
if ( ch = = ' = ' )
padding = true ;
else
{
if ( padding ) return false ; // other chars after padding
if ( ! i2p : : data : : IsBase64 ( ch ) ) return false ;
}
}
2023-10-29 22:11:38 -04:00
return true ;
}
return false ;
}
void HTTPReqHandler : : SanitizeHTTPRequest ( i2p : : http : : HTTPReq & req )
2016-07-01 00:00:00 +00:00
{
/* drop common headers */
2017-02-04 22:39:54 -05:00
req . RemoveHeader ( " Via " ) ;
2017-08-24 15:13:15 -04:00
req . RemoveHeader ( " From " ) ;
2017-09-03 09:46:55 -04:00
req . RemoveHeader ( " Forwarded " ) ;
2022-10-23 18:33:52 +00:00
req . RemoveHeader ( " DNT " ) ; // Useless DoNotTrack flag
2017-08-24 15:13:15 -04:00
req . RemoveHeader ( " Accept " , " Accept-Encoding " ) ; // Accept*, but Accept-Encoding
2016-07-01 00:00:00 +00:00
/* drop proxy-disclosing headers */
2017-02-04 22:39:54 -05:00
req . RemoveHeader ( " X-Forwarded " ) ;
2019-05-16 08:17:05 +03:00
req . RemoveHeader ( " Proxy- " ) ; // Proxy-*
2016-07-01 00:00:00 +00:00
/* replace headers */
2024-09-22 21:07:44 -04:00
if ( ! m_SendUserAgent )
req . UpdateHeader ( " User-Agent " , " MYOB/6.66 (AN/ON) " ) ;
2019-05-16 08:17:05 +03:00
2022-12-22 17:55:51 +00:00
/**
* i2pd PR # 1816 :
* Android Webview send this with the value set to the application ID , so we drop it ,
* but only if it does not belong to an AJAX request ( * HttpRequest , like XMLHttpRequest ) .
*/
if ( req . GetHeader ( " X-Requested-With " ) ! = " " ) {
auto h = req . GetHeader ( " X-Requested-With " ) ;
auto x = h . find ( " HttpRequest " ) ;
if ( x = = std : : string : : npos ) // not found
req . RemoveHeader ( " X-Requested-With " ) ;
}
2019-05-16 08:17:05 +03:00
/**
* according to i2p ticket # 1862 :
2022-05-27 13:29:59 -04:00
* leave Referer if requested URL with same schema , host and port ,
2019-05-16 08:17:05 +03:00
* otherwise , drop it .
*/
2022-05-27 13:29:59 -04:00
if ( req . GetHeader ( " Referer " ) ! = " " ) {
2019-05-16 08:17:05 +03:00
i2p : : http : : URL reqURL ; reqURL . parse ( req . uri ) ;
2022-05-27 13:29:59 -04:00
i2p : : http : : URL refURL ; refURL . parse ( req . GetHeader ( " Referer " ) ) ;
2019-05-16 08:17:05 +03:00
if ( ! boost : : iequals ( reqURL . schema , refURL . schema ) | | ! boost : : iequals ( reqURL . host , refURL . host ) | | reqURL . port ! = refURL . port )
2022-05-27 13:29:59 -04:00
req . RemoveHeader ( " Referer " ) ;
2019-05-16 08:17:05 +03:00
}
2017-02-04 22:39:54 -05:00
/* add headers */
2019-02-12 11:20:54 -05:00
/* close connection, if not Connection: (U|u)pgrade (for websocket) */
2019-02-11 17:18:01 -05:00
auto h = req . GetHeader ( " Connection " ) ;
2019-04-25 23:06:14 +03:00
auto x = h . find ( " pgrade " ) ;
2019-02-12 11:20:54 -05:00
if ( ! ( x ! = std : : string : : npos & & std : : tolower ( h [ x - 1 ] ) = = ' u ' ) )
req . UpdateHeader ( " Connection " , " close " ) ;
2016-07-01 00:00:00 +00:00
}
2016-07-01 00:00:00 +00:00
/**
* @ brief Try to parse request from @ a m_recv_buf
* If parsing success , rebuild request and store to @ a m_send_buf
* with remaining data tail
* @ return true on processed request or false if more data needed
*/
bool HTTPReqHandler : : HandleRequest ( )
2016-06-10 14:01:39 -04:00
{
2016-11-20 12:13:11 -05:00
m_req_len = m_ClientRequest . parse ( m_recv_buf ) ;
2016-07-01 00:00:00 +00:00
2016-11-20 12:13:11 -05:00
if ( m_req_len = = 0 )
2016-07-01 00:00:00 +00:00
return false ; /* need more data */
2016-11-20 12:13:11 -05:00
if ( m_req_len < 0 ) {
2021-11-27 22:53:53 +03:00
LogPrint ( eLogError , " HTTPProxy: Unable to parse request " ) ;
2021-05-23 06:06:04 +03:00
GenericProxyError ( tr ( " Invalid request " ) , tr ( " Proxy unable to parse your request " ) ) ;
2016-07-01 00:00:00 +00:00
return true ; /* parse error */
2016-07-01 00:00:00 +00:00
}
2016-07-01 00:00:00 +00:00
2016-07-01 00:00:00 +00:00
/* parsing success, now let's look inside request */
2021-11-27 22:53:53 +03:00
LogPrint ( eLogDebug , " HTTPProxy: Requested: " , m_ClientRequest . uri ) ;
2016-11-20 12:13:11 -05:00
m_RequestURL . parse ( m_ClientRequest . uri ) ;
2017-11-16 01:13:42 +03:00
bool m_Confirm ;
2016-07-01 00:00:00 +00:00
2019-04-02 13:11:49 -04:00
std : : string jump ;
if ( ExtractAddressHelper ( m_RequestURL , jump , m_Confirm ) )
2017-04-11 14:36:28 -04:00
{
2024-09-11 12:06:55 -04:00
if ( ! m_Addresshelper | | ! i2p : : client : : context . GetAddressBook ( ) . IsEnabled ( ) )
2017-04-11 14:36:28 -04:00
{
2021-11-27 22:53:53 +03:00
LogPrint ( eLogWarning , " HTTPProxy: Addresshelper request rejected " ) ;
2023-01-19 07:46:54 +03:00
GenericProxyError ( tr ( " Invalid request " ) , tr ( " Addresshelper is not supported " ) ) ;
2017-09-03 09:46:55 -04:00
return true ;
2017-04-11 14:36:28 -04:00
}
2022-10-09 22:13:49 +03:00
if ( i2p : : client : : context . GetAddressBook ( ) . RecordExists ( m_RequestURL . host , jump ) )
{
std : : string full_url = m_RequestURL . to_string ( ) ;
SendRedirect ( full_url ) ;
return true ;
}
else if ( ! i2p : : client : : context . GetAddressBook ( ) . FindAddress ( m_RequestURL . host ) | | m_Confirm )
2017-11-16 01:13:42 +03:00
{
2023-01-16 16:16:16 +03:00
const std : : string referer_raw = m_ClientRequest . GetHeader ( " Referer " ) ;
i2p : : http : : URL referer_url ;
if ( ! referer_raw . empty ( ) )
2023-01-16 13:31:13 +03:00
{
2023-01-16 16:16:16 +03:00
referer_url . parse ( referer_raw ) ;
}
if ( m_RequestURL . host ! = referer_url . host )
{
2023-01-19 07:46:54 +03:00
if ( m_Confirm ) // Attempt to forced overwriting by link with "&update=true" from harmful URL
2023-01-16 13:31:13 +03:00
{
2023-01-16 13:56:36 +03:00
LogPrint ( eLogWarning , " HTTPProxy: Address update from addresshelper rejected for " , m_RequestURL . host , " (referer is " , m_RequestURL . host . empty ( ) ? " empty " : " harmful " , " ) " ) ;
2023-01-16 13:31:13 +03:00
std : : string full_url = m_RequestURL . to_string ( ) ;
std : : stringstream ss ;
2023-01-19 07:46:54 +03:00
ss < < tr ( " Host %s is <font color=red>already in router's addressbook</font>. <b>Be careful: source of this URL may be harmful!</b> Click here to update record: <a href= \" %s%s%s&update=true \" >Continue</a>. " ,
2023-01-19 08:33:03 +03:00
m_RequestURL . host . c_str ( ) , full_url . c_str ( ) , ( full_url . find ( ' ? ' ) ! = std : : string : : npos ? " &i2paddresshelper= " : " ?i2paddresshelper= " ) , jump . c_str ( ) ) ;
2023-01-16 13:31:13 +03:00
GenericProxyInfo ( tr ( " Addresshelper forced update rejected " ) , ss . str ( ) ) ;
}
2023-01-19 07:46:54 +03:00
else // Preventing unauthorized additions to the address book
2023-01-16 16:16:16 +03:00
{
LogPrint ( eLogDebug , " HTTPProxy: Adding address from addresshelper for " , m_RequestURL . host , " (generate refer-base page) " ) ;
std : : string full_url = m_RequestURL . to_string ( ) ;
std : : stringstream ss ;
2023-01-19 07:46:54 +03:00
ss < < tr ( " To add host <b>%s</b> in router's addressbook, click here: <a href= \" %s%s%s \" >Continue</a>. " ,
2023-01-19 08:33:03 +03:00
m_RequestURL . host . c_str ( ) , full_url . c_str ( ) , ( full_url . find ( ' ? ' ) ! = std : : string : : npos ? " &i2paddresshelper= " : " ?i2paddresshelper= " ) , jump . c_str ( ) ) ;
2023-01-16 16:16:16 +03:00
GenericProxyInfo ( tr ( " Addresshelper request " ) , ss . str ( ) ) ;
}
return true ; /* request processed */
2023-01-16 13:31:13 +03:00
}
2019-04-02 13:11:49 -04:00
i2p : : client : : context . GetAddressBook ( ) . InsertAddress ( m_RequestURL . host , jump ) ;
2021-11-27 22:53:53 +03:00
LogPrint ( eLogInfo , " HTTPProxy: Added address from addresshelper for " , m_RequestURL . host ) ;
2017-11-16 01:13:42 +03:00
std : : string full_url = m_RequestURL . to_string ( ) ;
std : : stringstream ss ;
2023-01-19 08:33:03 +03:00
ss < < tr ( " Host %s added to router's addressbook from helper. Click here to proceed: <a href= \" %s \" >Continue</a>. " ,
m_RequestURL . host . c_str ( ) , full_url . c_str ( ) ) ;
2023-01-16 16:16:16 +03:00
GenericProxyInfo ( tr ( " Addresshelper adding " ) , ss . str ( ) ) ;
2017-11-16 01:13:42 +03:00
return true ; /* request processed */
}
else
{
2021-05-23 06:06:04 +03:00
std : : string full_url = m_RequestURL . to_string ( ) ;
2017-11-16 01:13:42 +03:00
std : : stringstream ss ;
2023-01-19 07:46:54 +03:00
ss < < tr ( " Host %s is <font color=red>already in router's addressbook</font>. Click here to update record: <a href= \" %s%s%s&update=true \" >Continue</a>. " ,
2023-01-19 08:33:03 +03:00
m_RequestURL . host . c_str ( ) , full_url . c_str ( ) , ( full_url . find ( ' ? ' ) ! = std : : string : : npos ? " &i2paddresshelper= " : " ?i2paddresshelper= " ) , jump . c_str ( ) ) ;
2023-01-16 16:16:16 +03:00
GenericProxyInfo ( tr ( " Addresshelper update " ) , ss . str ( ) ) ;
2017-11-16 01:13:42 +03:00
return true ; /* request processed */
}
2016-07-01 00:00:00 +00:00
}
2017-09-03 09:46:55 -04:00
std : : string dest_host ;
uint16_t dest_port ;
bool useConnect = false ;
if ( m_ClientRequest . method = = " CONNECT " )
2017-02-04 22:39:54 -05:00
{
2024-09-02 21:05:40 -04:00
const std : : string & uri = m_ClientRequest . uri ;
2017-09-03 09:46:55 -04:00
auto pos = uri . find ( " : " ) ;
if ( pos = = std : : string : : npos | | pos = = uri . size ( ) - 1 )
{
2023-01-19 07:46:54 +03:00
GenericProxyError ( tr ( " Invalid request " ) , tr ( " Invalid request URI " ) ) ;
2017-09-03 09:46:55 -04:00
return true ;
}
else
{
useConnect = true ;
dest_port = std : : stoi ( uri . substr ( pos + 1 ) ) ;
dest_host = uri . substr ( 0 , pos ) ;
}
}
2017-02-04 22:39:54 -05:00
else
2017-09-03 09:46:55 -04:00
{
SanitizeHTTPRequest ( m_ClientRequest ) ;
dest_host = m_RequestURL . host ;
dest_port = m_RequestURL . port ;
/* always set port, even if missing in request */
if ( ! dest_port )
dest_port = ( m_RequestURL . schema = = " https " ) ? 443 : 80 ;
/* detect dest_host, set proper 'Host' header in upstream request */
if ( dest_host ! = " " )
2017-02-04 22:39:54 -05:00
{
2017-09-03 09:46:55 -04:00
/* absolute url, replace 'Host' header */
2025-01-12 18:36:35 -05:00
std : : string h ( dest_host ) ;
2017-09-03 09:46:55 -04:00
if ( dest_port ! = 0 & & dest_port ! = 80 )
h + = " : " + std : : to_string ( dest_port ) ;
m_ClientRequest . UpdateHeader ( " Host " , h ) ;
}
else
2017-02-04 22:39:54 -05:00
{
2017-09-03 09:46:55 -04:00
auto h = m_ClientRequest . GetHeader ( " Host " ) ;
if ( h . length ( ) > 0 )
{
/* relative url and 'Host' header provided. transparent proxy mode? */
i2p : : http : : URL u ;
std : : string t = " http:// " + h ;
u . parse ( t ) ;
dest_host = u . host ;
dest_port = u . port ;
}
else
{
/* relative url and missing 'Host' header */
2021-05-23 06:06:04 +03:00
GenericProxyError ( tr ( " Invalid request " ) , tr ( " Can't detect destination host from request " ) ) ;
2017-09-03 09:46:55 -04:00
return true ;
}
2017-02-04 22:39:54 -05:00
}
2017-09-03 09:46:55 -04:00
}
2016-07-01 00:00:00 +00:00
/* check dest_host really exists and inside I2P network */
if ( str_rmatch ( dest_host , " .i2p " ) ) {
2019-03-28 12:19:19 -04:00
if ( ! i2p : : client : : context . GetAddressBook ( ) . GetAddress ( dest_host ) ) {
2016-07-14 00:00:00 +00:00
HostNotFound ( dest_host ) ;
2016-07-01 00:00:00 +00:00
return true ; /* request processed */
2016-03-26 02:45:37 -04:00
}
2016-07-01 00:00:00 +00:00
} else {
2017-10-27 08:42:54 -04:00
if ( m_OutproxyUrl . size ( ) ) {
2021-11-27 22:53:53 +03:00
LogPrint ( eLogDebug , " HTTPProxy: Using outproxy " , m_OutproxyUrl ) ;
2017-10-27 08:42:54 -04:00
if ( m_ProxyURL . parse ( m_OutproxyUrl ) )
2016-11-20 09:25:56 -05:00
ForwardToUpstreamProxy ( ) ;
else
2023-01-19 07:46:54 +03:00
GenericProxyError ( tr ( " Outproxy failure " ) , tr ( " Bad outproxy settings " ) ) ;
2016-11-20 09:25:56 -05:00
} else {
2021-11-27 22:53:53 +03:00
LogPrint ( eLogWarning , " HTTPProxy: Outproxy failure for " , dest_host , " : no outproxy enabled " ) ;
2025-01-12 18:36:35 -05:00
std : : stringstream ss ; ss < < tr ( " Host %s is not inside I2P network, but outproxy is not enabled " , dest_host . c_str ( ) ) ;
2021-05-23 06:06:04 +03:00
GenericProxyError ( tr ( " Outproxy failure " ) , ss . str ( ) ) ;
2016-11-20 09:25:56 -05:00
}
2016-07-01 00:00:00 +00:00
return true ;
2016-03-11 15:30:50 +05:00
}
2017-09-03 09:46:55 -04:00
if ( useConnect )
{
HTTPConnect ( dest_host , dest_port ) ;
return true ;
}
2016-06-10 14:01:39 -04:00
2016-07-01 00:00:00 +00:00
/* make relative url */
2016-11-20 09:25:56 -05:00
m_RequestURL . schema = " " ;
m_RequestURL . host = " " ;
2016-11-20 12:13:11 -05:00
m_ClientRequest . uri = m_RequestURL . to_string ( ) ;
2016-06-03 00:00:00 +00:00
2016-07-01 00:00:00 +00:00
/* drop original request from recv buffer */
2016-11-20 12:13:11 -05:00
m_recv_buf . erase ( 0 , m_req_len ) ;
2016-07-01 00:00:00 +00:00
/* build new buffer from modified request and data from original request */
2016-11-20 12:13:11 -05:00
m_send_buf = m_ClientRequest . to_string ( ) ;
2016-07-01 00:00:00 +00:00
m_send_buf . append ( m_recv_buf ) ;
/* connect to destination */
2021-11-27 22:53:53 +03:00
LogPrint ( eLogDebug , " HTTPProxy: Connecting to host " , dest_host , " : " , dest_port ) ;
2016-07-01 00:00:00 +00:00
GetOwner ( ) - > CreateStream ( std : : bind ( & HTTPReqHandler : : HandleStreamRequestComplete ,
shared_from_this ( ) , std : : placeholders : : _1 ) , dest_host , dest_port ) ;
2015-01-07 00:15:38 +01:00
return true ;
}
2016-11-20 09:25:56 -05:00
void HTTPReqHandler : : ForwardToUpstreamProxy ( )
{
2021-11-27 22:53:53 +03:00
LogPrint ( eLogDebug , " HTTPProxy: Forwarded to upstream " ) ;
2016-11-20 12:13:11 -05:00
2022-07-27 00:37:48 +03:00
/* build http request */
2016-11-20 12:13:11 -05:00
m_ClientRequestURL = m_RequestURL ;
LogPrint ( eLogDebug , " HTTPProxy: " , m_ClientRequestURL . host ) ;
m_ClientRequestURL . schema = " " ;
m_ClientRequestURL . host = " " ;
2018-07-10 17:39:21 +08:00
std : : string origURI = m_ClientRequest . uri ; // TODO: what do we need to change uri for?
2016-11-20 12:13:11 -05:00
m_ClientRequest . uri = m_ClientRequestURL . to_string ( ) ;
2024-09-22 21:27:09 -04:00
/* update User-Agent to ESR version of Firefox, same as Tor Browser below version 13, for non-HTTPS connections */
if ( m_ClientRequest . method ! = " CONNECT " & & ! m_SendUserAgent )
m_ClientRequest . UpdateHeader ( " User-Agent " , " Mozilla/5.0 (Windows NT 10.0; rv:109.0) Gecko/20100101 Firefox/115.0 " ) ;
2018-11-27 19:56:35 +03:00
2016-11-20 12:13:11 -05:00
m_ClientRequest . write ( m_ClientRequestBuffer ) ;
m_ClientRequestBuffer < < m_recv_buf . substr ( m_req_len ) ;
2019-04-25 23:06:14 +03:00
2022-07-27 00:37:48 +03:00
/* assume http if empty schema */
2019-04-25 23:06:14 +03:00
if ( m_ProxyURL . schema = = " " | | m_ProxyURL . schema = = " http " )
2018-04-06 15:23:56 -04:00
{
2022-07-27 00:37:48 +03:00
/* handle upstream http proxy */
2016-11-20 09:25:56 -05:00
if ( ! m_ProxyURL . port ) m_ProxyURL . port = 80 ;
2017-10-27 08:42:54 -04:00
if ( m_ProxyURL . is_i2p ( ) )
{
2018-04-13 16:29:49 -04:00
m_ClientRequest . uri = origURI ;
2021-03-10 22:47:31 -05:00
auto auth = i2p : : http : : CreateBasicAuthorizationString ( m_ProxyURL . user , m_ProxyURL . pass ) ;
if ( ! auth . empty ( ) )
2018-04-13 15:40:25 -04:00
{
2022-07-27 00:37:48 +03:00
/* remove existing authorization if any */
2018-04-13 15:40:25 -04:00
m_ClientRequest . RemoveHeader ( " Proxy- " ) ;
2022-07-27 00:37:48 +03:00
/* add own http proxy authorization */
2021-03-10 22:47:31 -05:00
m_ClientRequest . AddHeader ( " Proxy-Authorization " , auth ) ;
2018-04-13 15:40:25 -04:00
}
m_send_buf = m_ClientRequest . to_string ( ) ;
m_recv_buf . erase ( 0 , m_req_len ) ;
m_send_buf . append ( m_recv_buf ) ;
2017-10-27 08:42:54 -04:00
GetOwner ( ) - > CreateStream ( std : : bind ( & HTTPReqHandler : : HandleStreamRequestComplete ,
shared_from_this ( ) , std : : placeholders : : _1 ) , m_ProxyURL . host , m_ProxyURL . port ) ;
}
else
{
2024-11-25 16:00:06 -05:00
m_proxy_resolver . async_resolve ( m_ProxyURL . host , std : : to_string ( m_ProxyURL . port ) , std : : bind ( & HTTPReqHandler : : HandleUpstreamProxyResolved , this ,
std : : placeholders : : _1 , std : : placeholders : : _2 , [ & ] ( boost : : asio : : ip : : tcp : : endpoint ep )
{
m_proxysock - > async_connect ( ep , std : : bind ( & HTTPReqHandler : : HandleUpstreamHTTPProxyConnect , this , std : : placeholders : : _1 ) ) ;
} ) ) ;
2017-10-27 08:42:54 -04:00
}
2019-04-25 23:06:14 +03:00
}
else if ( m_ProxyURL . schema = = " socks " )
2018-04-06 15:23:56 -04:00
{
2022-07-27 00:37:48 +03:00
/* handle upstream socks proxy */
2016-11-20 09:25:56 -05:00
if ( ! m_ProxyURL . port ) m_ProxyURL . port = 9050 ; // default to tor default if not specified
2024-11-25 16:00:06 -05:00
m_proxy_resolver . async_resolve ( m_ProxyURL . host , std : : to_string ( m_ProxyURL . port ) , std : : bind ( & HTTPReqHandler : : HandleUpstreamProxyResolved , this ,
std : : placeholders : : _1 , std : : placeholders : : _2 , [ & ] ( boost : : asio : : ip : : tcp : : endpoint ep )
{
m_proxysock - > async_connect ( ep , std : : bind ( & HTTPReqHandler : : HandleUpstreamSocksProxyConnect , this , std : : placeholders : : _1 ) ) ;
} ) ) ;
2019-04-25 23:06:14 +03:00
}
else
2018-04-06 15:23:56 -04:00
{
2022-07-27 00:37:48 +03:00
/* unknown type, complain */
2023-01-19 07:46:54 +03:00
GenericProxyError ( tr ( " Unknown outproxy URL " ) , m_ProxyURL . to_string ( ) ) ;
2016-11-20 09:25:56 -05:00
}
}
2024-11-25 16:00:06 -05:00
void HTTPReqHandler : : HandleUpstreamProxyResolved ( const boost : : system : : error_code & ec , boost : : asio : : ip : : tcp : : resolver : : results_type endpoints , ProxyResolvedHandler handler )
2016-11-20 09:25:56 -05:00
{
2023-01-19 07:46:54 +03:00
if ( ec ) GenericProxyError ( tr ( " Cannot resolve upstream proxy " ) , ec . message ( ) ) ;
2024-11-25 16:00:06 -05:00
else handler ( * endpoints . begin ( ) ) ;
2016-11-20 09:25:56 -05:00
}
void HTTPReqHandler : : HandleUpstreamSocksProxyConnect ( const boost : : system : : error_code & ec )
{
2024-02-11 11:23:37 -05:00
if ( ! ec )
{
if ( m_RequestURL . host . size ( ) > 255 )
{
2023-01-19 07:46:54 +03:00
GenericProxyError ( tr ( " Hostname is too long " ) , m_RequestURL . host ) ;
2016-11-20 09:25:56 -05:00
return ;
}
uint16_t port = m_RequestURL . port ;
2016-11-20 12:13:11 -05:00
if ( ! port ) port = 80 ;
2021-11-27 22:53:53 +03:00
LogPrint ( eLogDebug , " HTTPProxy: Connected to SOCKS upstream " ) ;
2016-11-20 12:13:11 -05:00
std : : string host = m_RequestURL . host ;
2024-02-11 11:23:37 -05:00
auto s = shared_from_this ( ) ;
i2p : : transport : : Socks5Handshake ( * m_proxysock , std : : make_pair ( host , port ) ,
[ s ] ( const boost : : system : : error_code & ec )
{
if ( ! ec )
s - > SocksProxySuccess ( ) ;
else
s - > GenericProxyError ( tr ( " SOCKS proxy error " ) , ec . message ( ) ) ;
} ) ;
}
else
GenericProxyError ( tr ( " Cannot connect to upstream SOCKS proxy " ) , ec . message ( ) ) ;
2016-11-20 09:25:56 -05:00
}
void HTTPReqHandler : : HandoverToUpstreamProxy ( )
{
2021-11-27 22:53:53 +03:00
LogPrint ( eLogDebug , " HTTPProxy: Handover to SOCKS proxy " ) ;
2024-02-13 18:52:18 -05:00
auto connection = CreateSocketsPipe ( GetOwner ( ) , m_proxysock , m_sock ) ;
2016-11-20 12:13:11 -05:00
m_sock = nullptr ;
m_proxysock = nullptr ;
GetOwner ( ) - > AddHandler ( connection ) ;
connection - > Start ( ) ;
Terminate ( ) ;
2016-11-20 09:25:56 -05:00
}
2017-09-03 09:46:55 -04:00
2025-01-12 18:36:35 -05:00
void HTTPReqHandler : : HTTPConnect ( std : : string_view host , uint16_t port )
2017-09-03 09:46:55 -04:00
{
2017-09-03 11:13:43 -04:00
LogPrint ( eLogDebug , " HTTPProxy: CONNECT " , host , " : " , port ) ;
2025-01-12 18:36:35 -05:00
if ( str_rmatch ( host , " .i2p " ) )
2017-09-03 09:46:55 -04:00
GetOwner ( ) - > CreateStream ( std : : bind ( & HTTPReqHandler : : HandleHTTPConnectStreamRequestComplete ,
2019-04-25 23:06:14 +03:00
shared_from_this ( ) , std : : placeholders : : _1 ) , host , port ) ;
2017-09-03 09:46:55 -04:00
else
ForwardToUpstreamProxy ( ) ;
}
void HTTPReqHandler : : HandleHTTPConnectStreamRequestComplete ( std : : shared_ptr < i2p : : stream : : Stream > stream )
{
if ( stream )
{
2017-09-03 11:13:43 -04:00
m_ClientResponse . code = 200 ;
m_ClientResponse . status = " OK " ;
2017-09-03 09:46:55 -04:00
m_send_buf = m_ClientResponse . to_string ( ) ;
2017-09-03 11:13:43 -04:00
m_sock - > send ( boost : : asio : : buffer ( m_send_buf ) ) ;
auto connection = std : : make_shared < i2p : : client : : I2PTunnelConnection > ( GetOwner ( ) , m_sock , stream ) ;
GetOwner ( ) - > AddHandler ( connection ) ;
connection - > I2PConnect ( ) ;
m_sock = nullptr ;
Terminate ( ) ;
2017-09-03 09:46:55 -04:00
}
else
{
2023-01-19 07:46:54 +03:00
GenericProxyError ( tr ( " CONNECT error " ) , tr ( " Failed to connect " ) ) ;
2017-09-03 09:46:55 -04:00
}
}
2016-11-20 09:25:56 -05:00
void HTTPReqHandler : : SocksProxySuccess ( )
{
2016-11-20 12:13:11 -05:00
if ( m_ClientRequest . method = = " CONNECT " ) {
m_ClientResponse . code = 200 ;
m_send_buf = m_ClientResponse . to_string ( ) ;
2020-03-01 13:25:50 +03:00
boost : : asio : : async_write ( * m_sock , boost : : asio : : buffer ( m_send_buf ) , boost : : asio : : transfer_all ( ) , [ & ] ( const boost : : system : : error_code & ec , std : : size_t transferred )
{
2023-01-19 07:46:54 +03:00
if ( ec ) GenericProxyError ( tr ( " SOCKS proxy error " ) , ec . message ( ) ) ;
2016-11-20 12:13:11 -05:00
else HandoverToUpstreamProxy ( ) ;
} ) ;
} else {
m_send_buf = m_ClientRequestBuffer . str ( ) ;
2021-11-27 22:53:53 +03:00
LogPrint ( eLogDebug , " HTTPProxy: Send " , m_send_buf . size ( ) , " bytes " ) ;
2020-03-01 13:25:50 +03:00
boost : : asio : : async_write ( * m_proxysock , boost : : asio : : buffer ( m_send_buf ) , boost : : asio : : transfer_all ( ) , [ & ] ( const boost : : system : : error_code & ec , std : : size_t transferred )
{
2023-01-19 07:46:54 +03:00
if ( ec ) GenericProxyError ( tr ( " Failed to send request to upstream " ) , ec . message ( ) ) ;
2016-11-20 12:13:11 -05:00
else HandoverToUpstreamProxy ( ) ;
} ) ;
}
2016-11-20 09:25:56 -05:00
}
2017-09-03 09:46:55 -04:00
2016-11-20 09:25:56 -05:00
void HTTPReqHandler : : HandleUpstreamHTTPProxyConnect ( const boost : : system : : error_code & ec )
{
if ( ! ec ) {
2021-11-27 22:53:53 +03:00
LogPrint ( eLogDebug , " HTTPProxy: Connected to http upstream " ) ;
2023-01-19 07:46:54 +03:00
GenericProxyError ( tr ( " Cannot connect " ) , tr ( " HTTP out proxy not implemented " ) ) ;
} else GenericProxyError ( tr ( " Cannot connect to upstream HTTP proxy " ) , ec . message ( ) ) ;
2016-11-20 09:25:56 -05:00
}
2017-09-03 09:46:55 -04:00
2016-07-01 00:00:00 +00:00
/* will be called after some data received from client */
2016-06-23 00:00:00 +00:00
void HTTPReqHandler : : HandleSockRecv ( const boost : : system : : error_code & ecode , std : : size_t len )
2015-01-07 00:15:38 +01:00
{
2021-11-27 22:53:53 +03:00
LogPrint ( eLogDebug , " HTTPProxy: Sock recv: " , len , " bytes, recv buf: " , m_recv_buf . length ( ) , " , send buf: " , m_send_buf . length ( ) ) ;
2017-09-03 09:46:55 -04:00
if ( ecode )
2015-03-17 11:44:01 -04:00
{
2021-11-27 22:53:53 +03:00
LogPrint ( eLogWarning , " HTTPProxy: Sock recv got error: " , ecode ) ;
2016-02-11 00:00:00 +00:00
Terminate ( ) ;
2015-01-07 00:15:38 +01:00
return ;
}
2016-07-01 00:00:00 +00:00
m_recv_buf . append ( reinterpret_cast < const char * > ( m_recv_chunk ) , len ) ;
if ( HandleRequest ( ) ) {
m_recv_buf . clear ( ) ;
return ;
2016-06-04 00:00:00 +00:00
}
2016-07-01 00:00:00 +00:00
AsyncSockRead ( ) ;
2015-01-07 00:15:38 +01:00
}
2016-06-23 00:00:00 +00:00
void HTTPReqHandler : : SentHTTPFailed ( const boost : : system : : error_code & ecode )
2015-01-07 00:15:38 +01:00
{
2016-02-11 00:00:00 +00:00
if ( ecode )
LogPrint ( eLogError , " HTTPProxy: Closing socket after sending failure because: " , ecode . message ( ) ) ;
Terminate ( ) ;
2015-01-07 00:15:38 +01:00
}
2016-06-23 00:00:00 +00:00
void HTTPReqHandler : : HandleStreamRequestComplete ( std : : shared_ptr < i2p : : stream : : Stream > stream )
2015-01-07 00:15:38 +01:00
{
2016-06-27 13:00:00 +00:00
if ( ! stream ) {
2021-11-27 22:53:53 +03:00
LogPrint ( eLogError , " HTTPProxy: Error when creating the stream, check the previous warnings for more info " ) ;
2021-05-23 06:06:04 +03:00
GenericProxyError ( tr ( " Host is down " ) , tr ( " Can't create connection to requested host, it may be down. Please try again later. " ) ) ;
2016-06-27 13:00:00 +00:00
return ;
2015-01-07 00:15:38 +01:00
}
2016-06-27 13:00:00 +00:00
if ( Kill ( ) )
return ;
2016-07-01 00:00:00 +00:00
LogPrint ( eLogDebug , " HTTPProxy: Created new I2PTunnel stream, sSID= " , stream - > GetSendStreamID ( ) , " , rSID= " , stream - > GetRecvStreamID ( ) ) ;
2017-02-06 21:39:15 -05:00
auto connection = std : : make_shared < i2p : : client : : I2PClientTunnelConnectionHTTP > ( GetOwner ( ) , m_sock , stream ) ;
2016-06-27 13:00:00 +00:00
GetOwner ( ) - > AddHandler ( connection ) ;
2016-07-01 00:00:00 +00:00
connection - > I2PConnect ( reinterpret_cast < const uint8_t * > ( m_send_buf . data ( ) ) , m_send_buf . length ( ) ) ;
2016-06-27 13:00:00 +00:00
Done ( shared_from_this ( ) ) ;
2015-01-07 00:15:38 +01:00
}
2024-09-22 21:07:44 -04:00
HTTPProxy : : HTTPProxy ( const std : : string & name , const std : : string & address , uint16_t port ,
const std : : string & outproxy , bool addresshelper , bool senduseragent , std : : shared_ptr < i2p : : client : : ClientDestination > localDestination ) :
2019-04-25 23:06:14 +03:00
TCPIPAcceptor ( address , port , localDestination ? localDestination : i2p : : client : : context . GetSharedLocalDestination ( ) ) ,
2024-09-22 21:07:44 -04:00
m_Name ( name ) , m_OutproxyUrl ( outproxy ) , m_Addresshelper ( addresshelper ) , m_SendUserAgent ( senduseragent )
2015-01-17 09:42:44 -05:00
{
}
2017-09-03 09:46:55 -04:00
2016-06-23 00:00:00 +00:00
std : : shared_ptr < i2p : : client : : I2PServiceHandler > HTTPProxy : : CreateHandler ( std : : shared_ptr < boost : : asio : : ip : : tcp : : socket > socket )
2015-01-07 00:15:38 +01:00
{
2016-06-23 00:00:00 +00:00
return std : : make_shared < HTTPReqHandler > ( this , socket ) ;
2014-03-30 02:16:23 +04:00
}
2016-06-23 00:00:00 +00:00
} // http
} // i2p