2015-11-21 17:26:12 -05:00
# include <iomanip>
2016-04-27 00:00:00 +00:00
# include <sstream>
# include <thread>
# include <memory>
# include <boost/asio.hpp>
2013-12-10 08:03:22 -05:00
# include <boost/bind.hpp>
2016-04-27 00:00:00 +00:00
2015-11-03 09:15:49 -05:00
# include "Base.h"
2016-02-11 00:00:00 +00:00
# include "FS.h"
2014-01-14 19:56:34 -05:00
# include "Log.h"
2016-04-27 00:00:00 +00:00
# include "Config.h"
2013-12-10 08:03:22 -05:00
# include "Tunnel.h"
# include "Transports.h"
2017-04-21 20:04:16 -04:00
# include "NetDb.hpp"
2016-04-27 00:00:00 +00:00
# include "HTTP.h"
2016-04-27 00:00:00 +00:00
# include "LeaseSet.h"
2014-10-05 08:54:59 -04:00
# include "Destination.h"
2014-09-30 13:34:29 -04:00
# include "RouterContext.h"
2014-10-15 20:52:17 -04:00
# include "ClientContext.h"
2014-09-30 13:34:29 -04:00
# include "HTTPServer.h"
2016-04-27 00:00:00 +00:00
# include "Daemon.h"
2016-06-14 14:37:22 -04:00
# include "util.h"
2016-07-26 12:11:52 -04:00
# ifdef WIN32_APP
# include "Win32/Win32App.h"
# endif
2013-12-10 08:03:22 -05:00
2014-08-17 13:20:57 +02:00
// For image and info
# include "version.h"
2016-04-27 00:00:00 +00:00
namespace i2p {
namespace http {
2016-04-27 00:00:00 +00:00
const char * itoopieFavicon =
2016-01-12 13:18:01 -05:00
" data:image/png;base64, "
2016-11-21 00:58:38 +03:00
" iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAAAXNSR0IArs4c6QAAAARnQU1BAACx "
" jwv8YQUAAAAJcEhZcwAALiIAAC4iAari3ZIAAAAHdElNRQfgCQsUNSZrkhi1AAAAGXRFWHRTb2Z0 "
" d2FyZQBwYWludC5uZXQgNC4wLjEyQwRr7AAAAoJJREFUOE9jwAUqi4Q1oEwwcDTV1+5sETaBclGB "
" vb09C5QJB6kWpvFQJoOCeLC5kmjEHCgXE2SlyETLi3h6QrkM4VL+ssWSCZUgtopITLKqaOotRTEn "
" cbAkLqAkGtOqLBLVAWLXyWSVFkkmRiqLxuaqiWb/VBYJMAYrwgckJY25VEUzniqKhjU2y+RtCRSP "
" 6lUXy/1jIBV5tlYxZUaFVMq2NInwIi9hO8fSfOEAqDZUoCwal6MulvOvyS7gi69K4j9zxZT/m0ps "
" /28ptvvvquXXryIa7QYMMdTwqi0WNtVi0GIDseXl7TnUxFKfnGlxAGp0+D8j2eH/8Ub7/9e7nf7X "
" +Af/B7rwt6pI0h0l0WhQADOC9DBkhSirpImHNVZKp24ukkyoshGLnN8d5fA/y13t/44Kq/8hlnL/ "
" z7fZ/58f6vcxSNpbVUVFhV1RLNBVTsQzVYZPSwhsCAhkiIfpNMrkbO6TLf071Sfk/5ZSi/+7q6z/ "
" P5ns+v9mj/P/CpuI/20y+aeNGYxZoVoYGmsF3aFMBAAZlCwftnF9ke3//bU2//fXWP8/UGv731Am "
" +V+DdNblSqnUYqhSTKAiYSOqJBrVqiaa+S3UNPr/gmyH/xuKXf63hnn/B8bIP0UxHfEyyeSNQKVM "
" EB1AEB2twhcTLp+gIBJUoyKasEpVJHmqskh8qryovUG/ffCHHRU2q/Tk/YuB6eGPsbExa7ZkpLu1 "
" oLEcVDtuUCgV1w60rQzElpRUE1EVSX0BYidHiInXF4nagNhYQW60EF+ApH1ktni0A1SIITSUgVlZ "
" JHYnlIsfzJjIp9xZKswL5YKBHL+coKJoRDaUSzoozxHVrygQU4JykQADAwAT5b1NHtwZugAAAABJ "
" RU5ErkJggg== " ;
2014-07-16 18:41:40 +02:00
2016-04-27 00:00:00 +00:00
const char * cssStyles =
" <style> \r \n "
" body { font: 100%/1.5em sans-serif; margin: 0; padding: 1.5em; background: #FAFAFA; color: #103456; } \r \n "
" a { text-decoration: none; color: #894C84; } \r \n "
" a:hover { color: #FAFAFA; background: #894C84; } \r \n "
" .header { font-size: 2.5em; text-align: center; margin: 1.5em 0; color: #894C84; } \r \n "
" .wrapper { margin: 0 auto; padding: 1em; max-width: 60em; } \r \n "
" .left { float: left; position: absolute; } \r \n "
" .right { float: left; font-size: 1em; margin-left: 13em; max-width: 46em; overflow: auto; } \r \n "
" .tunnel.established { color: #56B734; } \r \n "
" .tunnel.expiring { color: #D3AE3F; } \r \n "
" .tunnel.failed { color: #D33F3F; } \r \n "
" .tunnel.another { color: #434343; } \r \n "
" caption { font-size: 1.5em; text-align: center; color: #894C84; } \r \n "
" table { width: 100%; border-collapse: collapse; text-align: center; } \r \n "
2016-06-30 07:35:30 -04:00
" .private { background: black; color: black; } .private:hover { background: black; color: white } \r \n "
2016-06-30 21:21:37 +03:00
" .slide p, .slide [type='checkbox']{ display:none; } \r \n "
" .slide [type='checkbox']:checked ~ p { display:block; } \r \n "
2016-04-27 00:00:00 +00:00
" </style> \r \n " ;
const char HTTP_PAGE_TUNNELS [ ] = " tunnels " ;
const char HTTP_PAGE_TRANSIT_TUNNELS [ ] = " transit_tunnels " ;
2016-08-09 13:54:47 +03:00
const char HTTP_PAGE_TRANSPORTS [ ] = " transports " ;
2016-04-27 00:00:00 +00:00
const char HTTP_PAGE_LOCAL_DESTINATIONS [ ] = " local_destinations " ;
const char HTTP_PAGE_LOCAL_DESTINATION [ ] = " local_destination " ;
2016-11-17 19:14:25 -05:00
const char HTTP_PAGE_I2CP_LOCAL_DESTINATION [ ] = " i2cp_local_destination " ;
2016-04-27 00:00:00 +00:00
const char HTTP_PAGE_SAM_SESSIONS [ ] = " sam_sessions " ;
const char HTTP_PAGE_SAM_SESSION [ ] = " sam_session " ;
const char HTTP_PAGE_I2P_TUNNELS [ ] = " i2p_tunnels " ;
2016-04-27 00:00:00 +00:00
const char HTTP_PAGE_COMMANDS [ ] = " commands " ;
2016-07-15 13:52:55 -04:00
const char HTTP_PAGE_LEASESETS [ ] = " leasesets " ;
2016-07-12 00:00:00 +00:00
const char HTTP_COMMAND_ENABLE_TRANSIT [ ] = " enable_transit " ;
const char HTTP_COMMAND_DISABLE_TRANSIT [ ] = " disable_transit " ;
2016-04-27 00:00:00 +00:00
const char HTTP_COMMAND_SHUTDOWN_START [ ] = " shutdown_start " ;
const char HTTP_COMMAND_SHUTDOWN_CANCEL [ ] = " shutdown_cancel " ;
const char HTTP_COMMAND_SHUTDOWN_NOW [ ] = " terminate " ;
2016-05-12 11:38:18 -04:00
const char HTTP_COMMAND_RUN_PEER_TEST [ ] = " run_peer_test " ;
2016-08-09 13:54:47 +03:00
const char HTTP_COMMAND_RELOAD_CONFIG [ ] = " reload_config " ;
2015-02-20 22:47:36 -05:00
const char HTTP_PARAM_SAM_SESSION_ID [ ] = " id " ;
2016-03-11 15:30:50 +05:00
const char HTTP_PARAM_ADDRESS [ ] = " address " ;
2013-12-10 08:03:22 -05:00
2016-11-17 19:14:25 -05:00
static void ShowUptime ( std : : stringstream & s , int seconds )
{
2016-04-27 00:00:00 +00:00
int num ;
2013-12-10 08:03:22 -05:00
2016-04-27 00:00:00 +00:00
if ( ( num = seconds / 86400 ) > 0 ) {
s < < num < < " days, " ;
2016-05-19 00:00:00 +00:00
seconds - = num * 86400 ;
2013-12-10 08:03:22 -05:00
}
2016-04-27 00:00:00 +00:00
if ( ( num = seconds / 3600 ) > 0 ) {
s < < num < < " hours, " ;
2016-05-19 00:00:00 +00:00
seconds - = num * 3600 ;
2014-07-16 18:41:40 +02:00
}
2016-04-27 00:00:00 +00:00
if ( ( num = seconds / 60 ) > 0 ) {
s < < num < < " min, " ;
2016-05-19 00:00:00 +00:00
seconds - = num * 60 ;
2016-04-27 00:00:00 +00:00
}
s < < seconds < < " seconds " ;
2014-03-30 02:16:23 +04:00
}
2017-04-27 16:11:37 -04:00
static void ShowTraffic ( std : : stringstream & s , uint64_t bytes )
{
s < < std : : fixed < < std : : setprecision ( 2 ) ;
auto numKBytes = ( double ) bytes / 1024 ;
if ( numKBytes < 1024 )
s < < numKBytes < < " KiB " ;
else if ( numKBytes < 1024 * 1024 )
s < < numKBytes / 1024 < < " MiB " ;
else
s < < numKBytes / 1024 / 1024 < < " GiB " ;
}
2016-11-17 19:14:25 -05:00
static void ShowTunnelDetails ( std : : stringstream & s , enum i2p : : tunnel : : TunnelState eState , int bytes )
2014-01-14 19:56:34 -05:00
{
2016-04-27 00:00:00 +00:00
std : : string state ;
switch ( eState ) {
case i2p : : tunnel : : eTunnelStateBuildReplyReceived :
case i2p : : tunnel : : eTunnelStatePending : state = " building " ; break ;
case i2p : : tunnel : : eTunnelStateBuildFailed :
case i2p : : tunnel : : eTunnelStateTestFailed :
case i2p : : tunnel : : eTunnelStateFailed : state = " failed " ; break ;
case i2p : : tunnel : : eTunnelStateExpiring : state = " expiring " ; break ;
case i2p : : tunnel : : eTunnelStateEstablished : state = " established " ; break ;
default : state = " unknown " ; break ;
2014-07-16 18:41:40 +02:00
}
2016-04-27 00:00:00 +00:00
s < < " <span class= \" tunnel " < < state < < " \" > " < < state < < " </span>, " ;
s < < " " < < ( int ) ( bytes / 1024 ) < < " KiB<br> \r \n " ;
2014-07-16 18:41:40 +02:00
}
2016-11-17 19:14:25 -05:00
static void ShowPageHead ( std : : stringstream & s )
2014-03-27 15:42:23 -04:00
{
2016-04-27 00:00:00 +00:00
s < <
" <!DOCTYPE html> \r \n "
" <html lang= \" en \" > \r \n " /* TODO: Add support for locale */
2016-06-28 00:00:54 +03:00
" <head> \r \n " /* TODO: Find something to parse html/template system. This is horrible. */
# if (!defined(WIN32))
" <meta charset= \" UTF-8 \" > \r \n "
# else
" <meta charset= \" windows-1251 \" > \r \n "
# endif
2016-06-14 11:25:51 +03:00
" <link rel= \" shortcut icon \" href= \" " < < itoopieFavicon < < " \" > \r \n "
2016-04-27 00:00:00 +00:00
" <title>Purple I2P " VERSION " Webconsole</title> \r \n "
< < cssStyles < <
" </head> \r \n " ;
s < <
" <body> \r \n "
" <div class=header><b>i2pd</b> webconsole</div> \r \n "
" <div class=wrapper> \r \n "
" <div class=left> \r \n "
2016-06-14 11:25:51 +03:00
" <a href= \" / \" >Main page</a><br> \r \n <br> \r \n "
" <a href= \" /?page= " < < HTTP_PAGE_COMMANDS < < " \" >Router commands</a><br> \r \n "
" <a href= \" /?page= " < < HTTP_PAGE_LOCAL_DESTINATIONS < < " \" >Local destinations</a><br> \r \n "
2016-08-09 13:54:47 +03:00
" <a href= \" /?page= " < < HTTP_PAGE_LEASESETS < < " \" >LeaseSets</a><br> \r \n "
2016-06-14 11:25:51 +03:00
" <a href= \" /?page= " < < HTTP_PAGE_TUNNELS < < " \" >Tunnels</a><br> \r \n "
" <a href= \" /?page= " < < HTTP_PAGE_TRANSIT_TUNNELS < < " \" >Transit tunnels</a><br> \r \n "
" <a href= \" /?page= " < < HTTP_PAGE_TRANSPORTS < < " \" >Transports</a><br> \r \n "
2016-08-09 13:54:47 +03:00
" <a href= \" /?page= " < < HTTP_PAGE_I2P_TUNNELS < < " \" >I2P tunnels</a><br> \r \n " ;
if ( i2p : : client : : context . GetSAMBridge ( ) )
s < < " <a href= \" /?page= " < < HTTP_PAGE_SAM_SESSIONS < < " \" >SAM sessions</a><br> \r \n " ;
s < <
2016-04-27 00:00:00 +00:00
" </div> \r \n "
" <div class=right> " ;
2014-03-27 15:42:23 -04:00
}
2016-11-17 19:14:25 -05:00
static void ShowPageTail ( std : : stringstream & s )
2013-12-10 08:03:22 -05:00
{
2016-04-27 00:00:00 +00:00
s < <
" </div></div> \r \n "
" </body> \r \n "
" </html> \r \n " ;
2013-12-10 08:03:22 -05:00
}
2016-11-17 19:14:25 -05:00
static void ShowError ( std : : stringstream & s , const std : : string & string )
2013-12-10 08:03:22 -05:00
{
2016-04-27 00:00:00 +00:00
s < < " <b>ERROR:</b> " < < string < < " <br> \r \n " ;
2014-07-16 18:41:40 +02:00
}
2013-12-10 08:03:22 -05:00
2017-08-02 02:40:11 +08:00
void ShowStatus ( std : : stringstream & s , bool includeHiddenContent )
2013-12-10 08:03:22 -05:00
{
2016-04-27 00:00:00 +00:00
s < < " <b>Uptime:</b> " ;
ShowUptime ( s , i2p : : context . GetUptime ( ) ) ;
s < < " <br> \r \n " ;
2016-08-21 00:00:00 +00:00
s < < " <b>Network status:</b> " ;
2015-02-26 13:44:18 -05:00
switch ( i2p : : context . GetStatus ( ) )
{
case eRouterStatusOK : s < < " OK " ; break ;
case eRouterStatusTesting : s < < " Testing " ; break ;
case eRouterStatusFirewalled : s < < " Firewalled " ; break ;
2016-09-19 21:37:04 -04:00
case eRouterStatusError :
{
s < < " Error " ;
switch ( i2p : : context . GetError ( ) )
{
case eRouterErrorClockSkew :
s < < " <br>Clock skew " ;
break ;
default : ;
}
break ;
}
2015-02-26 13:44:18 -05:00
default : s < < " Unknown " ;
}
2016-01-17 11:10:56 -05:00
s < < " <br> \r \n " ;
2016-08-21 00:00:00 +00:00
# if (!defined(WIN32) && !defined(QT_GUI_LIB) && !defined(ANDROID))
2016-10-31 03:27:27 -07:00
if ( auto remains = Daemon . gracefulShutdownInterval ) {
2016-08-21 00:00:00 +00:00
s < < " <b>Stopping in:</b> " ;
s < < remains < < " seconds " ;
s < < " <br> \r \n " ;
}
# endif
2016-04-28 18:16:11 -04:00
auto family = i2p : : context . GetFamily ( ) ;
if ( family . length ( ) > 0 )
s < < " <b>Family:</b> " < < family < < " <br> \r \n " ;
2016-01-17 11:10:56 -05:00
s < < " <b>Tunnel creation success rate:</b> " < < i2p : : tunnel : : tunnels . GetTunnelCreationSuccessRate ( ) < < " %<br> \r \n " ;
2016-03-26 10:32:19 -04:00
s < < " <b>Received:</b> " ;
2017-04-27 16:11:37 -04:00
ShowTraffic ( s , i2p : : transport : : transports . GetTotalReceivedBytes ( ) ) ;
2016-03-26 10:32:19 -04:00
s < < " ( " < < ( double ) i2p : : transport : : transports . GetInBandwidth ( ) / 1024 < < " KiB/s)<br> \r \n " ;
s < < " <b>Sent:</b> " ;
2017-04-27 16:11:37 -04:00
ShowTraffic ( s , i2p : : transport : : transports . GetTotalSentBytes ( ) ) ;
2016-03-26 10:32:19 -04:00
s < < " ( " < < ( double ) i2p : : transport : : transports . GetOutBandwidth ( ) / 1024 < < " KiB/s)<br> \r \n " ;
2017-04-27 16:11:37 -04:00
s < < " <b>Transit:</b> " ;
2017-05-02 14:20:00 -04:00
ShowTraffic ( s , i2p : : transport : : transports . GetTotalTransitTransmittedBytes ( ) ) ;
s < < " ( " < < ( double ) i2p : : transport : : transports . GetTransitBandwidth ( ) / 1024 < < " KiB/s)<br> \r \n " ;
2016-11-12 17:49:16 +03:00
s < < " <b>Data path:</b> " < < i2p : : fs : : GetDataDir ( ) < < " <br> \r \n " ;
2016-06-30 21:21:37 +03:00
s < < " <div class='slide' \r \n ><label for='slide1'>Hidden content. Press on text to see.</label> \r \n <input type='checkbox' id='slide1'/> \r \n <p class='content'> \r \n " ;
2017-08-02 02:40:11 +08:00
if ( includeHiddenContent ) {
s < < " <b>Router Ident:</b> " < < i2p : : context . GetRouterInfo ( ) . GetIdentHashBase64 ( ) < < " <br> \r \n " ;
s < < " <b>Router Family:</b> " < < i2p : : context . GetRouterInfo ( ) . GetProperty ( " family " ) < < " <br> \r \n " ;
s < < " <b>Router Caps:</b> " < < i2p : : context . GetRouterInfo ( ) . GetProperty ( " caps " ) < < " <br> \r \n " ;
s < < " <b>Our external address:</b> " < < " <br> \r \n " ;
for ( const auto & address : i2p : : context . GetRouterInfo ( ) . GetAddresses ( ) )
{
switch ( address - > transportStyle )
{
case i2p : : data : : RouterInfo : : eTransportNTCP :
if ( address - > host . is_v6 ( ) )
s < < " NTCP6 " ;
else
s < < " NTCP " ;
break ;
case i2p : : data : : RouterInfo : : eTransportSSU :
if ( address - > host . is_v6 ( ) )
s < < " SSU6 " ;
else
s < < " SSU " ;
break ;
default :
s < < " Unknown " ;
}
s < < address - > host . to_string ( ) < < " : " < < address - > port < < " <br> \r \n " ;
}
}
s < < " </p> \r \n </div> \r \n " ;
s < < " <b>Routers:</b> " < < i2p : : data : : netdb . GetNumRouters ( ) < < " " ;
2016-01-11 22:05:10 -05:00
s < < " <b>Floodfills:</b> " < < i2p : : data : : netdb . GetNumFloodfills ( ) < < " " ;
2016-01-17 11:10:56 -05:00
s < < " <b>LeaseSets:</b> " < < i2p : : data : : netdb . GetNumLeaseSets ( ) < < " <br> \r \n " ;
2016-03-02 09:41:37 -05:00
2016-03-02 10:05:26 -05:00
size_t clientTunnelCount = i2p : : tunnel : : tunnels . CountOutboundTunnels ( ) ;
clientTunnelCount + = i2p : : tunnel : : tunnels . CountInboundTunnels ( ) ;
size_t transitTunnelCount = i2p : : tunnel : : tunnels . CountTransitTunnels ( ) ;
2016-08-09 13:54:47 +03:00
s < < " <b>Client Tunnels:</b> " < < std : : to_string ( clientTunnelCount ) < < " " ;
s < < " <b>Transit Tunnels:</b> " < < std : : to_string ( transitTunnelCount ) < < " <br> \r \n " ;
2014-09-28 16:12:25 -04:00
}
2017-08-02 04:12:37 +08:00
void ShowLocalDestinations ( std : : stringstream & s )
2014-09-28 19:15:04 -04:00
{
2016-01-17 11:10:56 -05:00
s < < " <b>Local Destinations:</b><br> \r \n <br> \r \n " ;
2014-10-15 20:52:17 -04:00
for ( auto & it : i2p : : client : : context . GetDestinations ( ) )
2014-09-28 19:15:04 -04:00
{
2016-11-17 19:14:25 -05:00
auto ident = it . second - > GetIdentHash ( ) ;
2016-06-14 11:25:51 +03:00
s < < " <a href= \" /?page= " < < HTTP_PAGE_LOCAL_DESTINATION < < " &b32= " < < ident . ToBase32 ( ) < < " \" > " ;
2016-01-17 11:10:56 -05:00
s < < i2p : : client : : context . GetAddressBook ( ) . ToAddress ( ident ) < < " </a><br> \r \n " < < std : : endl ;
2014-09-28 19:15:04 -04:00
}
2016-11-17 19:14:25 -05:00
auto i2cpServer = i2p : : client : : context . GetI2CPServer ( ) ;
if ( i2cpServer )
{
s < < " <br><b>I2CP Local Destinations:</b><br> \r \n <br> \r \n " ;
for ( auto & it : i2cpServer - > GetSessions ( ) )
{
auto dest = it . second - > GetDestination ( ) ;
if ( dest )
{
auto ident = dest - > GetIdentHash ( ) ;
s < < " <a href= \" /?page= " < < HTTP_PAGE_I2CP_LOCAL_DESTINATION < < " &i2cp_id= " < < it . first < < " \" > " ;
s < < i2p : : client : : context . GetAddressBook ( ) . ToAddress ( ident ) < < " </a><br> \r \n " < < std : : endl ;
}
}
}
}
static void ShowLeaseSetDestination ( std : : stringstream & s , std : : shared_ptr < const i2p : : client : : LeaseSetDestination > dest )
{
s < < " <b>Base64:</b><br> \r \n <textarea readonly= \" readonly \" cols= \" 64 \" rows= \" 11 \" wrap= \" on \" > " ;
s < < dest - > GetIdentity ( ) - > ToBase64 ( ) < < " </textarea><br> \r \n <br> \r \n " ;
s < < " <b>LeaseSets:</b> <i> " < < dest - > GetNumRemoteLeaseSets ( ) < < " </i><br> \r \n " ;
if ( dest - > GetNumRemoteLeaseSets ( ) )
{
s < < " <div class='slide' \r \n ><label for='slide1'>Hidden content. Press on text to see.</label> \r \n <input type='checkbox' id='slide1'/> \r \n <p class='content'> \r \n " ;
for ( auto & it : dest - > GetLeaseSets ( ) )
s < < it . second - > GetIdentHash ( ) . ToBase32 ( ) < < " <br> \r \n " ;
s < < " </p> \r \n </div> \r \n " ;
}
auto pool = dest - > GetTunnelPool ( ) ;
if ( pool )
{
s < < " <b>Inbound tunnels:</b><br> \r \n " ;
for ( auto & it : pool - > GetInboundTunnels ( ) ) {
it - > Print ( s ) ;
if ( it - > LatencyIsKnown ( ) )
s < < " ( " < < it - > GetMeanLatency ( ) < < " ms ) " ;
ShowTunnelDetails ( s , it - > GetState ( ) , it - > GetNumReceivedBytes ( ) ) ;
}
s < < " <br> \r \n " ;
s < < " <b>Outbound tunnels:</b><br> \r \n " ;
for ( auto & it : pool - > GetOutboundTunnels ( ) ) {
it - > Print ( s ) ;
if ( it - > LatencyIsKnown ( ) )
s < < " ( " < < it - > GetMeanLatency ( ) < < " ms ) " ;
ShowTunnelDetails ( s , it - > GetState ( ) , it - > GetNumSentBytes ( ) ) ;
}
}
s < < " <br> \r \n " ;
s < < " <b>Tags</b><br>Incoming: " < < dest - > GetNumIncomingTags ( ) < < " <br>Outgoing:<br> " < < std : : endl ;
for ( const auto & it : dest - > GetSessions ( ) )
{
s < < i2p : : client : : context . GetAddressBook ( ) . ToAddress ( it . first ) < < " " ;
s < < it . second - > GetNumOutgoingTags ( ) < < " <br> " < < std : : endl ;
}
s < < " <br> " < < std : : endl ;
2016-01-12 20:31:25 -05:00
}
2014-09-29 22:18:32 -04:00
2017-08-20 10:22:30 +08:00
void ShowLocalDestination ( std : : stringstream & s , const std : : string & b32 )
2014-09-29 22:18:32 -04:00
{
2016-01-17 11:10:56 -05:00
s < < " <b>Local Destination:</b><br> \r \n <br> \r \n " ;
2014-09-29 22:18:32 -04:00
i2p : : data : : IdentHash ident ;
2015-04-10 12:11:10 -04:00
ident . FromBase32 ( b32 ) ;
2014-10-16 10:28:44 -04:00
auto dest = i2p : : client : : context . FindLocalDestination ( ident ) ;
2014-09-29 22:18:32 -04:00
if ( dest )
{
2016-11-17 19:14:25 -05:00
ShowLeaseSetDestination ( s , dest ) ;
// show streams
2016-01-17 11:10:56 -05:00
s < < " <br> \r \n <table><caption>Streams</caption><tr> " ;
s < < " <th>StreamID</th> " ;
s < < " <th>Destination</th> " ;
s < < " <th>Sent</th> " ;
s < < " <th>Received</th> " ;
s < < " <th>Out</th> " ;
s < < " <th>In</th> " ;
s < < " <th>Buf</th> " ;
s < < " <th>RTT</th> " ;
s < < " <th>Window</th> " ;
s < < " <th>Status</th> " ;
s < < " </tr> " ;
2016-08-05 21:23:54 +03:00
for ( const auto & it : dest - > GetAllStreams ( ) )
2016-08-09 13:54:47 +03:00
{
2016-01-17 11:10:56 -05:00
s < < " <tr> " ;
2016-04-04 22:17:04 -04:00
s < < " <td> " < < it - > GetSendStreamID ( ) < < " </td> " ;
s < < " <td> " < < i2p : : client : : context . GetAddressBook ( ) . ToAddress ( it - > GetRemoteIdentity ( ) ) < < " </td> " ;
s < < " <td> " < < it - > GetNumSentBytes ( ) < < " </td> " ;
s < < " <td> " < < it - > GetNumReceivedBytes ( ) < < " </td> " ;
s < < " <td> " < < it - > GetSendQueueSize ( ) < < " </td> " ;
s < < " <td> " < < it - > GetReceiveQueueSize ( ) < < " </td> " ;
s < < " <td> " < < it - > GetSendBufferSize ( ) < < " </td> " ;
s < < " <td> " < < it - > GetRTT ( ) < < " </td> " ;
s < < " <td> " < < it - > GetWindowSize ( ) < < " </td> " ;
s < < " <td> " < < ( int ) it - > GetStatus ( ) < < " </td> " ;
2016-01-17 11:10:56 -05:00
s < < " </tr><br> \r \n " < < std : : endl ;
2016-09-03 13:58:34 -04:00
}
2016-09-03 18:04:54 -04:00
s < < " </table> " ;
2016-08-09 13:54:47 +03:00
}
2016-01-12 20:31:25 -05:00
}
2016-11-17 19:14:25 -05:00
static void ShowI2CPLocalDestination ( std : : stringstream & s , const std : : string & id )
{
auto i2cpServer = i2p : : client : : context . GetI2CPServer ( ) ;
if ( i2cpServer )
{
s < < " <b>I2CP Local Destination:</b><br> \r \n <br> \r \n " ;
auto it = i2cpServer - > GetSessions ( ) . find ( std : : stoi ( id ) ) ;
if ( it ! = i2cpServer - > GetSessions ( ) . end ( ) )
ShowLeaseSetDestination ( s , it - > second - > GetDestination ( ) ) ;
else
ShowError ( s , " I2CP session not found " ) ;
}
else
ShowError ( s , " I2CP is not enabled " ) ;
}
2017-08-02 04:12:37 +08:00
void ShowLeasesSets ( std : : stringstream & s )
2016-07-15 13:52:55 -04:00
{
2016-09-19 20:22:15 +03:00
s < < " <div id='leasesets'><b>LeaseSets (click on to show info):</b></div><br> \r \n " ;
int counter = 1 ;
2016-07-15 13:52:55 -04:00
// for each lease set
i2p : : data : : netdb . VisitLeaseSets (
2016-09-19 20:22:15 +03:00
[ & s , & counter ] ( const i2p : : data : : IdentHash dest , std : : shared_ptr < i2p : : data : : LeaseSet > leaseSet )
2016-07-15 13:52:55 -04:00
{
// create copy of lease set so we extract leases
i2p : : data : : LeaseSet ls ( leaseSet - > GetBuffer ( ) , leaseSet - > GetBufferLen ( ) ) ;
s < < " <div class='leaseset " ;
if ( ls . IsExpired ( ) )
s < < " expired " ; // additional css class for expired
2016-09-19 20:22:15 +03:00
s < < " '> \r \n " ;
2016-07-15 13:52:55 -04:00
if ( ! ls . IsValid ( ) )
2016-09-19 20:22:15 +03:00
s < < " <div class='invalid'>!! Invalid !! </div> \r \n " ;
s < < " <div class='slide'><label for='slide " < < counter < < " '> " < < dest . ToBase32 ( ) < < " </label> \r \n " ;
s < < " <input type='checkbox' id='slide " < < ( counter + + ) < < " '/> \r \n <p class='content'> \r \n " ;
s < < " <b>Expires:</b> " < < ls . GetExpirationTime ( ) < < " <br> \r \n " ;
2016-07-15 13:52:55 -04:00
auto leases = ls . GetNonExpiredLeases ( ) ;
2016-09-19 20:22:15 +03:00
s < < " <b>Non Expired Leases: " < < leases . size ( ) < < " </b><br> \r \n " ;
2016-07-15 13:52:55 -04:00
for ( auto & l : leases )
{
2016-09-19 20:22:15 +03:00
s < < " <b>Gateway:</b> " < < l - > tunnelGateway . ToBase64 ( ) < < " <br> \r \n " ;
s < < " <b>TunnelID:</b> " < < l - > tunnelID < < " <br> \r \n " ;
s < < " <b>EndDate:</b> " < < l - > endDate < < " <br> \r \n " ;
2016-07-15 13:52:55 -04:00
}
2016-09-19 20:22:15 +03:00
s < < " </p> \r \n </div> \r \n </div> \r \n " ;
2016-07-15 13:52:55 -04:00
}
) ;
// end for each lease set
}
2016-08-09 13:54:47 +03:00
2017-08-02 04:12:37 +08:00
void ShowTunnels ( std : : stringstream & s )
2016-01-12 20:31:25 -05:00
{
2016-01-17 18:18:21 +00:00
s < < " <b>Queue size:</b> " < < i2p : : tunnel : : tunnels . GetQueueSize ( ) < < " <br> \r \n " ;
2016-01-12 20:31:25 -05:00
2016-04-27 00:00:00 +00:00
s < < " <b>Inbound tunnels:</b><br> \r \n " ;
for ( auto & it : i2p : : tunnel : : tunnels . GetInboundTunnels ( ) ) {
it - > Print ( s ) ;
2016-11-15 14:42:18 -05:00
if ( it - > LatencyIsKnown ( ) )
s < < " ( " < < it - > GetMeanLatency ( ) < < " ms ) " ;
2016-04-27 00:00:00 +00:00
ShowTunnelDetails ( s , it - > GetState ( ) , it - > GetNumReceivedBytes ( ) ) ;
2016-01-12 20:31:25 -05:00
}
2016-04-27 00:00:00 +00:00
s < < " <br> \r \n " ;
s < < " <b>Outbound tunnels:</b><br> \r \n " ;
for ( auto & it : i2p : : tunnel : : tunnels . GetOutboundTunnels ( ) ) {
it - > Print ( s ) ;
2016-11-15 14:42:18 -05:00
if ( it - > LatencyIsKnown ( ) )
s < < " ( " < < it - > GetMeanLatency ( ) < < " ms ) " ;
2016-04-27 00:00:00 +00:00
ShowTunnelDetails ( s , it - > GetState ( ) , it - > GetNumSentBytes ( ) ) ;
2016-01-12 20:31:25 -05:00
}
2016-04-27 00:00:00 +00:00
s < < " <br> \r \n " ;
2016-08-09 13:54:47 +03:00
}
2015-02-20 22:47:36 -05:00
2017-08-02 05:05:56 +08:00
static void ShowCommands ( std : : stringstream & s , uint32_t token )
2016-04-27 00:00:00 +00:00
{
/* commands */
s < < " <b>Router Commands</b><br> \r \n " ;
2016-12-15 13:10:12 -05:00
s < < " <a href= \" /?cmd= " < < HTTP_COMMAND_RUN_PEER_TEST < < " &token= " < < token < < " \" >Run peer test</a><br> \r \n " ;
2016-06-14 11:25:51 +03:00
//s << " <a href=\"/?cmd=" << HTTP_COMMAND_RELOAD_CONFIG << "\">Reload config</a><br>\r\n";
2016-04-27 00:00:00 +00:00
if ( i2p : : context . AcceptsTunnels ( ) )
2016-12-15 13:10:12 -05:00
s < < " <a href= \" /?cmd= " < < HTTP_COMMAND_DISABLE_TRANSIT < < " &token= " < < token < < " \" >Decline transit tunnels</a><br> \r \n " ;
2016-08-09 13:54:47 +03:00
else
2016-12-15 13:10:12 -05:00
s < < " <a href= \" /?cmd= " < < HTTP_COMMAND_ENABLE_TRANSIT < < " &token= " < < token < < " \" >Accept transit tunnels</a><br> \r \n " ;
2016-07-10 04:54:11 +08:00
# if (!defined(WIN32) && !defined(QT_GUI_LIB) && !defined(ANDROID))
2016-10-31 03:27:27 -07:00
if ( Daemon . gracefulShutdownInterval )
2016-12-15 13:10:12 -05:00
s < < " <a href= \" /?cmd= " < < HTTP_COMMAND_SHUTDOWN_CANCEL < < " &token= " < < token < < " \" >Cancel graceful shutdown</a><br> " ;
2016-07-26 12:11:52 -04:00
else
2016-12-15 13:10:12 -05:00
s < < " <a href= \" /?cmd= " < < HTTP_COMMAND_SHUTDOWN_START < < " &token= " < < token < < " \" >Start graceful shutdown</a><br> \r \n " ;
2016-05-11 15:33:53 -04:00
# endif
2016-07-26 12:11:52 -04:00
# ifdef WIN32_APP
2016-12-15 13:10:12 -05:00
s < < " <a href= \" /?cmd= " < < HTTP_COMMAND_SHUTDOWN_START < < " &token= " < < token < < " \" >Graceful shutdown</a><br> \r \n " ;
2016-07-26 12:11:52 -04:00
# endif
2016-12-15 13:10:12 -05:00
s < < " <a href= \" /?cmd= " < < HTTP_COMMAND_SHUTDOWN_NOW < < " &token= " < < token < < " \" >Force shutdown</a><br> \r \n " ;
2016-04-27 00:00:00 +00:00
}
2017-08-02 04:12:37 +08:00
void ShowTransitTunnels ( std : : stringstream & s )
2016-01-12 20:31:25 -05:00
{
2016-01-17 11:10:56 -05:00
s < < " <b>Transit tunnels:</b><br> \r \n <br> \r \n " ;
2016-08-05 21:23:54 +03:00
for ( const auto & it : i2p : : tunnel : : tunnels . GetTransitTunnels ( ) )
2016-01-12 20:31:25 -05:00
{
2016-03-01 20:48:56 -05:00
if ( std : : dynamic_pointer_cast < i2p : : tunnel : : TransitTunnelGateway > ( it ) )
2016-08-09 13:54:47 +03:00
s < < it - > GetTunnelID ( ) < < " ⇒ " ;
2016-03-01 20:48:56 -05:00
else if ( std : : dynamic_pointer_cast < i2p : : tunnel : : TransitTunnelEndpoint > ( it ) )
2016-08-09 13:54:47 +03:00
s < < " ⇒ " < < it - > GetTunnelID ( ) ;
2016-01-12 20:31:25 -05:00
else
2016-08-09 13:54:47 +03:00
s < < " ⇒ " < < it - > GetTunnelID ( ) < < " ⇒ " ;
2016-03-01 20:48:56 -05:00
s < < " " < < it - > GetNumTransmittedBytes ( ) < < " <br> \r \n " ;
2016-01-12 20:31:25 -05:00
}
}
2017-08-02 04:12:37 +08:00
void ShowTransports ( std : : stringstream & s )
2016-01-12 20:31:25 -05:00
{
2016-01-17 11:10:56 -05:00
s < < " <b>Transports:</b><br> \r \n <br> \r \n " ;
2016-01-12 20:31:25 -05:00
auto ntcpServer = i2p : : transport : : transports . GetNTCPServer ( ) ;
if ( ntcpServer )
2016-08-09 13:54:47 +03:00
{
2017-01-01 08:58:21 -05:00
auto sessions = ntcpServer - > GetNTCPSessions ( ) ;
2017-07-08 16:54:12 +03:00
if ( ! sessions . empty ( ) )
2016-01-12 20:31:25 -05:00
{
2017-07-08 16:54:12 +03:00
std : : stringstream tmp_s , tmp_s6 ; uint16_t cnt = 0 , cnt6 = 0 ;
for ( const auto & it : sessions )
2016-01-12 20:31:25 -05:00
{
2017-07-08 16:54:12 +03:00
if ( it . second & & it . second - > IsEstablished ( ) & & ! it . second - > GetSocket ( ) . remote_endpoint ( ) . address ( ) . is_v6 ( ) )
{
// incoming connection doesn't have remote RI
if ( it . second - > IsOutgoing ( ) ) tmp_s < < " ⇒ " ;
tmp_s < < i2p : : data : : GetIdentHashAbbreviation ( it . second - > GetRemoteIdentity ( ) - > GetIdentHash ( ) ) < < " : "
< < it . second - > GetSocket ( ) . remote_endpoint ( ) . address ( ) . to_string ( ) ;
if ( ! it . second - > IsOutgoing ( ) ) tmp_s < < " ⇒ " ;
tmp_s < < " [ " < < it . second - > GetNumSentBytes ( ) < < " : " < < it . second - > GetNumReceivedBytes ( ) < < " ] " ;
tmp_s < < " <br> \r \n " < < std : : endl ;
cnt + + ;
}
if ( it . second & & it . second - > IsEstablished ( ) & & it . second - > GetSocket ( ) . remote_endpoint ( ) . address ( ) . is_v6 ( ) )
{
if ( it . second - > IsOutgoing ( ) ) tmp_s6 < < " ⇒ " ;
tmp_s6 < < i2p : : data : : GetIdentHashAbbreviation ( it . second - > GetRemoteIdentity ( ) - > GetIdentHash ( ) ) < < " : "
< < " [ " < < it . second - > GetSocket ( ) . remote_endpoint ( ) . address ( ) . to_string ( ) < < " ] " ;
if ( ! it . second - > IsOutgoing ( ) ) tmp_s6 < < " ⇒ " ;
tmp_s6 < < " [ " < < it . second - > GetNumSentBytes ( ) < < " : " < < it . second - > GetNumReceivedBytes ( ) < < " ] " ;
tmp_s6 < < " <br> \r \n " < < std : : endl ;
cnt6 + + ;
}
}
if ( ! tmp_s . str ( ) . empty ( ) )
{
s < < " <b>NTCP</b> ( " < < cnt < < " )<br> \r \n " ;
s < < tmp_s . str ( ) < < " <br> \r \n " ;
}
if ( ! tmp_s6 . str ( ) . empty ( ) )
{
s < < " <b>NTCP6</b> ( " < < cnt6 < < " )<br> \r \n " ;
s < < tmp_s6 . str ( ) < < " <br> \r \n " ;
2016-01-12 20:31:25 -05:00
}
}
2016-08-09 13:54:47 +03:00
}
2016-01-12 20:31:25 -05:00
auto ssuServer = i2p : : transport : : transports . GetSSUServer ( ) ;
if ( ssuServer )
{
2017-01-01 08:58:21 -05:00
auto sessions = ssuServer - > GetSessions ( ) ;
2017-07-08 16:54:12 +03:00
if ( ! sessions . empty ( ) )
2016-01-12 20:31:25 -05:00
{
2017-07-08 16:54:12 +03:00
s < < " <b>SSU</b> ( " < < ( int ) sessions . size ( ) < < " )<br> \r \n " ;
for ( const auto & it : sessions )
{
auto endpoint = it . second - > GetRemoteEndpoint ( ) ;
if ( it . second - > IsOutgoing ( ) ) s < < " ⇒ " ;
s < < endpoint . address ( ) . to_string ( ) < < " : " < < endpoint . port ( ) ;
if ( ! it . second - > IsOutgoing ( ) ) s < < " ⇒ " ;
s < < " [ " < < it . second - > GetNumSentBytes ( ) < < " : " < < it . second - > GetNumReceivedBytes ( ) < < " ] " ;
if ( it . second - > GetRelayTag ( ) )
s < < " [itag: " < < it . second - > GetRelayTag ( ) < < " ] " ;
s < < " <br> \r \n " < < std : : endl ;
}
2017-07-08 16:59:10 +03:00
s < < " <br> \r \n " ;
2016-01-12 20:31:25 -05:00
}
2017-07-08 16:54:12 +03:00
auto sessions6 = ssuServer - > GetSessionsV6 ( ) ;
if ( ! sessions6 . empty ( ) )
2016-01-12 20:31:25 -05:00
{
2017-07-08 16:54:12 +03:00
s < < " <b>SSU6</b> ( " < < ( int ) sessions6 . size ( ) < < " )<br> \r \n " ;
for ( const auto & it : sessions6 )
{
auto endpoint = it . second - > GetRemoteEndpoint ( ) ;
if ( it . second - > IsOutgoing ( ) ) s < < " ⇒ " ;
s < < " [ " < < endpoint . address ( ) . to_string ( ) < < " ]: " < < endpoint . port ( ) ;
if ( ! it . second - > IsOutgoing ( ) ) s < < " ⇒ " ;
s < < " [ " < < it . second - > GetNumSentBytes ( ) < < " : " < < it . second - > GetNumReceivedBytes ( ) < < " ] " ;
if ( it . second - > GetRelayTag ( ) )
s < < " [itag: " < < it . second - > GetRelayTag ( ) < < " ] " ;
s < < " <br> \r \n " < < std : : endl ;
}
2017-07-08 16:59:10 +03:00
s < < " <br> \r \n " ;
2016-01-12 20:31:25 -05:00
}
}
}
2016-08-09 13:54:47 +03:00
2017-08-02 04:12:37 +08:00
void ShowSAMSessions ( std : : stringstream & s )
2015-02-20 22:47:36 -05:00
{
auto sam = i2p : : client : : context . GetSAMBridge ( ) ;
2016-04-27 00:00:00 +00:00
if ( ! sam ) {
ShowError ( s , " SAM disabled " ) ;
return ;
}
s < < " <b>SAM Sessions:</b><br> \r \n <br> \r \n " ;
for ( auto & it : sam - > GetSessions ( ) )
{
2016-06-14 11:25:51 +03:00
s < < " <a href= \" /?page= " < < HTTP_PAGE_SAM_SESSION < < " &sam_id= " < < it . first < < " \" > " ;
2016-04-27 00:00:00 +00:00
s < < it . first < < " </a><br> \r \n " < < std : : endl ;
2016-08-09 13:54:47 +03:00
}
}
2015-02-20 22:47:36 -05:00
2016-11-17 19:14:25 -05:00
static void ShowSAMSession ( std : : stringstream & s , const std : : string & id )
2015-02-20 22:47:36 -05:00
{
2016-01-17 11:10:56 -05:00
s < < " <b>SAM Session:</b><br> \r \n <br> \r \n " ;
2015-02-20 22:47:36 -05:00
auto sam = i2p : : client : : context . GetSAMBridge ( ) ;
2016-04-27 00:00:00 +00:00
if ( ! sam ) {
ShowError ( s , " SAM disabled " ) ;
return ;
}
auto session = sam - > FindSession ( id ) ;
if ( ! session ) {
ShowError ( s , " SAM session not found " ) ;
return ;
}
auto & ident = session - > localDestination - > GetIdentHash ( ) ;
2016-06-14 11:25:51 +03:00
s < < " <a href= \" /?page= " < < HTTP_PAGE_LOCAL_DESTINATION < < " &b32= " < < ident . ToBase32 ( ) < < " \" > " ;
2016-04-27 00:00:00 +00:00
s < < i2p : : client : : context . GetAddressBook ( ) . ToAddress ( ident ) < < " </a><br> \r \n " ;
s < < " <br> \r \n " ;
s < < " <b>Streams:</b><br> \r \n " ;
2016-08-05 21:23:54 +03:00
for ( const auto & it : session - > ListSockets ( ) )
2015-02-20 22:47:36 -05:00
{
2016-04-27 00:00:00 +00:00
switch ( it - > GetSocketType ( ) )
2015-02-20 22:47:36 -05:00
{
2016-04-27 00:00:00 +00:00
case i2p : : client : : eSAMSocketTypeSession : s < < " session " ; break ;
case i2p : : client : : eSAMSocketTypeStream : s < < " stream " ; break ;
case i2p : : client : : eSAMSocketTypeAcceptor : s < < " acceptor " ; break ;
default : s < < " unknown " ; break ;
2015-02-20 22:47:36 -05:00
}
2016-04-27 00:00:00 +00:00
s < < " [ " < < it - > GetSocket ( ) . remote_endpoint ( ) < < " ] " ;
s < < " <br> \r \n " ;
2016-08-09 13:54:47 +03:00
}
}
2014-09-30 13:34:29 -04:00
2017-08-02 04:12:37 +08:00
void ShowI2PTunnels ( std : : stringstream & s )
2016-01-13 20:21:53 -05:00
{
2016-01-17 11:10:56 -05:00
s < < " <b>Client Tunnels:</b><br> \r \n <br> \r \n " ;
2016-01-13 20:21:53 -05:00
for ( auto & it : i2p : : client : : context . GetClientTunnels ( ) )
{
auto & ident = it . second - > GetLocalDestination ( ) - > GetIdentHash ( ) ;
2016-06-14 11:25:51 +03:00
s < < " <a href= \" /?page= " < < HTTP_PAGE_LOCAL_DESTINATION < < " &b32= " < < ident . ToBase32 ( ) < < " \" > " ;
2016-08-09 13:54:47 +03:00
s < < it . second - > GetName ( ) < < " </a> ⇐ " ;
2016-01-13 20:21:53 -05:00
s < < i2p : : client : : context . GetAddressBook ( ) . ToAddress ( ident ) ;
2016-04-06 16:33:23 -04:00
s < < " <br> \r \n " < < std : : endl ;
2016-08-09 13:54:47 +03:00
}
2016-10-31 15:42:50 -04:00
auto httpProxy = i2p : : client : : context . GetHttpProxy ( ) ;
if ( httpProxy )
{
auto & ident = httpProxy - > GetLocalDestination ( ) - > GetIdentHash ( ) ;
s < < " <a href= \" /?page= " < < HTTP_PAGE_LOCAL_DESTINATION < < " &b32= " < < ident . ToBase32 ( ) < < " \" > " ;
s < < " HTTP Proxy " < < " </a> ⇐ " ;
s < < i2p : : client : : context . GetAddressBook ( ) . ToAddress ( ident ) ;
s < < " <br> \r \n " < < std : : endl ;
}
2016-12-27 22:45:51 -05:00
auto socksProxy = i2p : : client : : context . GetSocksProxy ( ) ;
if ( socksProxy )
{
auto & ident = socksProxy - > GetLocalDestination ( ) - > GetIdentHash ( ) ;
s < < " <a href= \" /?page= " < < HTTP_PAGE_LOCAL_DESTINATION < < " &b32= " < < ident . ToBase32 ( ) < < " \" > " ;
s < < " SOCKS Proxy " < < " </a> ⇐ " ;
s < < i2p : : client : : context . GetAddressBook ( ) . ToAddress ( ident ) ;
s < < " <br> \r \n " < < std : : endl ;
2017-08-25 02:37:01 +03:00
}
auto & serverTunnels = i2p : : client : : context . GetServerTunnels ( ) ;
if ( ! serverTunnels . empty ( ) ) {
s < < " <br> \r \n <b>Server Tunnels:</b><br> \r \n <br> \r \n " ;
for ( auto & it : serverTunnels )
{
auto & ident = it . second - > GetLocalDestination ( ) - > GetIdentHash ( ) ;
s < < " <a href= \" /?page= " < < HTTP_PAGE_LOCAL_DESTINATION < < " &b32= " < < ident . ToBase32 ( ) < < " \" > " ;
s < < it . second - > GetName ( ) < < " </a> ⇒ " ;
s < < i2p : : client : : context . GetAddressBook ( ) . ToAddress ( ident ) ;
s < < " : " < < it . second - > GetLocalPort ( ) ;
s < < " </a><br> \r \n " < < std : : endl ;
}
2016-08-09 13:54:47 +03:00
}
2016-09-12 12:05:01 -04:00
auto & clientForwards = i2p : : client : : context . GetClientForwards ( ) ;
if ( ! clientForwards . empty ( ) )
{
s < < " <br> \r \n <b>Client Forwards:</b><br> \r \n <br> \r \n " ;
for ( auto & it : clientForwards )
{
auto & ident = it . second - > GetLocalDestination ( ) - > GetIdentHash ( ) ;
s < < " <a href= \" /?page= " < < HTTP_PAGE_LOCAL_DESTINATION < < " &b32= " < < ident . ToBase32 ( ) < < " \" > " ;
s < < it . second - > GetName ( ) < < " </a> ⇐ " ;
s < < i2p : : client : : context . GetAddressBook ( ) . ToAddress ( ident ) ;
s < < " <br> \r \n " < < std : : endl ;
}
}
auto & serverForwards = i2p : : client : : context . GetServerForwards ( ) ;
if ( ! serverForwards . empty ( ) )
{
s < < " <br> \r \n <b>Server Forwards:</b><br> \r \n <br> \r \n " ;
for ( auto & it : serverForwards )
{
auto & ident = it . second - > GetLocalDestination ( ) - > GetIdentHash ( ) ;
s < < " <a href= \" /?page= " < < HTTP_PAGE_LOCAL_DESTINATION < < " &b32= " < < ident . ToBase32 ( ) < < " \" > " ;
s < < it . second - > GetName ( ) < < " </a> ⇐ " ;
s < < i2p : : client : : context . GetAddressBook ( ) . ToAddress ( ident ) ;
s < < " <br> \r \n " < < std : : endl ;
}
}
2016-08-09 13:54:47 +03:00
}
2014-09-30 13:34:29 -04:00
2016-04-27 00:00:00 +00:00
HTTPConnection : : HTTPConnection ( std : : shared_ptr < boost : : asio : : ip : : tcp : : socket > socket ) :
m_Socket ( socket ) , m_Timer ( socket - > get_io_service ( ) ) , m_BufferLen ( 0 )
2016-01-12 20:31:25 -05:00
{
2016-04-27 00:00:00 +00:00
/* cache options */
i2p : : config : : GetOption ( " http.auth " , needAuth ) ;
i2p : : config : : GetOption ( " http.user " , user ) ;
i2p : : config : : GetOption ( " http.pass " , pass ) ;
2016-06-01 00:00:00 +00:00
}
2016-01-12 20:31:25 -05:00
2016-04-27 00:00:00 +00:00
void HTTPConnection : : Receive ( )
2015-11-03 09:15:49 -05:00
{
2016-04-27 00:00:00 +00:00
m_Socket - > async_read_some ( boost : : asio : : buffer ( m_Buffer , HTTP_CONNECTION_BUFFER_SIZE ) ,
std : : bind ( & HTTPConnection : : HandleReceive , shared_from_this ( ) ,
std : : placeholders : : _1 , std : : placeholders : : _2 ) ) ;
2015-11-03 09:15:49 -05:00
}
2016-04-27 00:00:00 +00:00
void HTTPConnection : : HandleReceive ( const boost : : system : : error_code & ecode , std : : size_t bytes_transferred )
2014-01-12 21:41:25 -05:00
{
2016-04-27 00:00:00 +00:00
if ( ecode ) {
if ( ecode ! = boost : : asio : : error : : operation_aborted )
Terminate ( ecode ) ;
2014-10-03 14:22:32 -04:00
return ;
2014-10-21 14:28:56 -04:00
}
2016-04-27 00:00:00 +00:00
m_Buffer [ bytes_transferred ] = ' \0 ' ;
m_BufferLen = bytes_transferred ;
RunRequest ( ) ;
Receive ( ) ;
2014-10-21 14:28:56 -04:00
}
2016-04-27 00:00:00 +00:00
void HTTPConnection : : RunRequest ( )
2014-10-21 14:28:56 -04:00
{
2016-04-27 00:00:00 +00:00
HTTPReq request ;
int ret = request . parse ( m_Buffer ) ;
if ( ret < 0 ) {
m_Buffer [ 0 ] = ' \0 ' ;
m_BufferLen = 0 ;
return ; /* error */
2014-07-16 18:41:40 +02:00
}
2016-04-27 00:00:00 +00:00
if ( ret = = 0 )
return ; /* need more data */
2016-04-27 00:00:00 +00:00
2016-04-27 00:00:00 +00:00
HandleRequest ( request ) ;
2014-10-21 14:28:56 -04:00
}
2016-04-27 00:00:00 +00:00
void HTTPConnection : : Terminate ( const boost : : system : : error_code & ecode )
2014-03-27 15:42:23 -04:00
{
2016-04-27 00:00:00 +00:00
if ( ecode = = boost : : asio : : error : : operation_aborted )
return ;
boost : : system : : error_code ignored_ec ;
m_Socket - > shutdown ( boost : : asio : : ip : : tcp : : socket : : shutdown_both , ignored_ec ) ;
m_Socket - > close ( ) ;
2014-03-27 15:42:23 -04:00
}
2016-04-27 00:00:00 +00:00
bool HTTPConnection : : CheckAuth ( const HTTPReq & req ) {
/* method #1: http://user:pass@127.0.0.1:7070/ */
if ( req . uri . find ( ' @ ' ) ! = std : : string : : npos ) {
URL url ;
if ( url . parse ( req . uri ) & & url . user = = user & & url . pass = = pass )
return true ;
2014-03-27 13:24:23 -04:00
}
2016-04-27 00:00:00 +00:00
/* method #2: 'Authorization' header sent */
2017-02-04 22:39:54 -05:00
auto provided = req . GetHeader ( " Authorization " ) ;
if ( provided . length ( ) > 0 )
{
2017-01-01 08:52:36 -05:00
bool result = false ;
2017-02-04 22:39:54 -05:00
2016-04-27 00:00:00 +00:00
std : : string expected = user + " : " + pass ;
2017-01-01 08:54:11 -05:00
size_t b64_sz = i2p : : data : : Base64EncodingBufferSize ( expected . length ( ) ) + 1 ;
char * b64_creds = new char [ b64_sz ] ;
2016-04-27 00:00:00 +00:00
std : : size_t len = 0 ;
2017-01-01 08:54:11 -05:00
len = i2p : : data : : ByteStreamToBase64 ( ( unsigned char * ) expected . c_str ( ) , expected . length ( ) , b64_creds , b64_sz ) ;
2016-12-09 09:27:19 -05:00
/* if we decoded properly then check credentials */
if ( len ) {
b64_creds [ len ] = ' \0 ' ;
expected = " Basic " ;
expected + = b64_creds ;
2017-01-01 08:52:36 -05:00
result = expected = = provided ;
2016-12-09 09:27:19 -05:00
}
2017-01-01 08:52:36 -05:00
delete [ ] b64_creds ;
return result ;
2014-07-16 18:41:40 +02:00
}
2016-04-27 00:00:00 +00:00
LogPrint ( eLogWarning , " HTTPServer: auth failure from " , m_Socket - > remote_endpoint ( ) . address ( ) ) ;
return false ;
2014-03-27 13:24:23 -04:00
}
2016-04-27 00:00:00 +00:00
void HTTPConnection : : HandleRequest ( const HTTPReq & req )
2014-03-27 13:24:23 -04:00
{
2016-04-27 00:00:00 +00:00
std : : stringstream s ;
2016-04-27 00:00:00 +00:00
std : : string content ;
HTTPRes res ;
2016-04-27 00:00:00 +00:00
2016-04-27 00:00:00 +00:00
LogPrint ( eLogDebug , " HTTPServer: request: " , req . uri ) ;
2016-04-27 00:00:00 +00:00
if ( needAuth & & ! CheckAuth ( req ) ) {
res . code = 401 ;
2016-05-24 00:00:00 +00:00
res . add_header ( " WWW-Authenticate " , " Basic realm= \" WebAdmin \" " ) ;
2016-04-27 00:00:00 +00:00
SendReply ( res , content ) ;
return ;
}
2016-04-27 00:00:00 +00:00
// Html5 head start
2016-04-27 00:00:00 +00:00
ShowPageHead ( s ) ;
2016-05-28 00:00:00 +00:00
if ( req . uri . find ( " page= " ) ! = std : : string : : npos ) {
2016-04-27 00:00:00 +00:00
HandlePage ( req , res , s ) ;
2016-05-28 00:00:00 +00:00
} else if ( req . uri . find ( " cmd= " ) ! = std : : string : : npos ) {
2016-04-27 00:00:00 +00:00
HandleCommand ( req , res , s ) ;
2016-05-28 00:00:00 +00:00
} else {
2017-08-02 02:40:11 +08:00
ShowStatus ( s , true ) ;
2016-06-24 19:07:47 -04:00
res . add_header ( " Refresh " , " 10 " ) ;
2016-05-28 00:00:00 +00:00
}
2016-04-27 00:00:00 +00:00
ShowPageTail ( s ) ;
2016-05-24 00:00:00 +00:00
res . code = 200 ;
2016-04-27 00:00:00 +00:00
content = s . str ( ) ;
SendReply ( res , content ) ;
2014-03-26 15:45:08 -04:00
}
2014-07-16 18:41:40 +02:00
2016-12-15 13:10:12 -05:00
std : : map < uint32_t , uint32_t > HTTPConnection : : m_Tokens ;
2016-04-27 00:00:00 +00:00
void HTTPConnection : : HandlePage ( const HTTPReq & req , HTTPRes & res , std : : stringstream & s )
2016-04-27 00:00:00 +00:00
{
std : : map < std : : string , std : : string > params ;
std : : string page ( " " ) ;
URL url ;
2016-04-27 00:00:00 +00:00
url . parse ( req . uri ) ;
2016-04-27 00:00:00 +00:00
url . parse_query ( params ) ;
page = params [ " page " ] ;
if ( page = = HTTP_PAGE_TRANSPORTS )
ShowTransports ( s ) ;
else if ( page = = HTTP_PAGE_TUNNELS )
ShowTunnels ( s ) ;
2016-04-27 00:00:00 +00:00
else if ( page = = HTTP_PAGE_COMMANDS )
2016-12-15 13:10:12 -05:00
{
uint32_t token ;
RAND_bytes ( ( uint8_t * ) & token , 4 ) ;
2016-12-22 20:34:06 -05:00
token & = 0x7FFFFFFF ; // clear first bit
2016-12-15 13:10:12 -05:00
auto ts = i2p : : util : : GetSecondsSinceEpoch ( ) ;
for ( auto it = m_Tokens . begin ( ) ; it ! = m_Tokens . end ( ) ; )
{
if ( ts > it - > second + TOKEN_EXPIRATION_TIMEOUT )
it = m_Tokens . erase ( it ) ;
else
+ + it ;
}
m_Tokens [ token ] = ts ;
ShowCommands ( s , token ) ;
}
2016-04-27 00:00:00 +00:00
else if ( page = = HTTP_PAGE_TRANSIT_TUNNELS )
ShowTransitTunnels ( s ) ;
else if ( page = = HTTP_PAGE_LOCAL_DESTINATIONS )
ShowLocalDestinations ( s ) ;
else if ( page = = HTTP_PAGE_LOCAL_DESTINATION )
ShowLocalDestination ( s , params [ " b32 " ] ) ;
2016-11-17 19:14:25 -05:00
else if ( page = = HTTP_PAGE_I2CP_LOCAL_DESTINATION )
ShowI2CPLocalDestination ( s , params [ " i2cp_id " ] ) ;
2016-04-27 00:00:00 +00:00
else if ( page = = HTTP_PAGE_SAM_SESSIONS )
ShowSAMSessions ( s ) ;
else if ( page = = HTTP_PAGE_SAM_SESSION )
ShowSAMSession ( s , params [ " sam_id " ] ) ;
else if ( page = = HTTP_PAGE_I2P_TUNNELS )
ShowI2PTunnels ( s ) ;
2016-07-15 13:52:55 -04:00
else if ( page = = HTTP_PAGE_LEASESETS )
ShowLeasesSets ( s ) ;
2016-04-27 00:00:00 +00:00
else {
res . code = 400 ;
ShowError ( s , " Unknown page: " + page ) ;
return ;
}
2016-04-27 00:00:00 +00:00
}
2016-04-27 00:00:00 +00:00
void HTTPConnection : : HandleCommand ( const HTTPReq & req , HTTPRes & res , std : : stringstream & s )
2016-04-27 00:00:00 +00:00
{
std : : map < std : : string , std : : string > params ;
URL url ;
2016-04-27 00:00:00 +00:00
url . parse ( req . uri ) ;
2016-04-27 00:00:00 +00:00
url . parse_query ( params ) ;
2016-12-15 13:10:12 -05:00
std : : string token = params [ " token " ] ;
2016-12-15 13:36:52 -05:00
if ( token . empty ( ) | | m_Tokens . find ( std : : stoi ( token ) ) = = m_Tokens . end ( ) )
2016-12-15 13:10:12 -05:00
{
ShowError ( s , " Invalid token " ) ;
return ;
}
std : : string cmd = params [ " cmd " ] ;
2016-04-27 00:00:00 +00:00
if ( cmd = = HTTP_COMMAND_RUN_PEER_TEST )
i2p : : transport : : transports . PeerTest ( ) ;
2016-05-12 11:38:18 -04:00
else if ( cmd = = HTTP_COMMAND_RELOAD_CONFIG )
i2p : : client : : context . ReloadConfig ( ) ;
2016-07-12 00:00:00 +00:00
else if ( cmd = = HTTP_COMMAND_ENABLE_TRANSIT )
2016-04-27 00:00:00 +00:00
i2p : : context . SetAcceptsTunnels ( true ) ;
2016-07-12 00:00:00 +00:00
else if ( cmd = = HTTP_COMMAND_DISABLE_TRANSIT )
2016-04-27 00:00:00 +00:00
i2p : : context . SetAcceptsTunnels ( false ) ;
2016-04-27 00:00:00 +00:00
else if ( cmd = = HTTP_COMMAND_SHUTDOWN_START ) {
i2p : : context . SetAcceptsTunnels ( false ) ;
2016-07-10 04:54:11 +08:00
# if (!defined(WIN32) && !defined(QT_GUI_LIB) && !defined(ANDROID))
2016-10-31 03:27:27 -07:00
Daemon . gracefulShutdownInterval = 10 * 60 ;
2016-07-26 12:11:52 -04:00
# endif
# ifdef WIN32_APP
2016-08-09 13:54:47 +03:00
i2p : : win32 : : GracefulShutdown ( ) ;
2016-05-11 15:33:53 -04:00
# endif
2016-04-27 00:00:00 +00:00
} else if ( cmd = = HTTP_COMMAND_SHUTDOWN_CANCEL ) {
i2p : : context . SetAcceptsTunnels ( true ) ;
2016-07-10 04:54:11 +08:00
# if (!defined(WIN32) && !defined(QT_GUI_LIB) && !defined(ANDROID))
2016-10-31 03:27:27 -07:00
Daemon . gracefulShutdownInterval = 0 ;
2016-05-11 15:33:53 -04:00
# endif
2016-04-27 00:00:00 +00:00
} else if ( cmd = = HTTP_COMMAND_SHUTDOWN_NOW ) {
Daemon . running = false ;
} else {
2016-04-27 00:00:00 +00:00
res . code = 400 ;
ShowError ( s , " Unknown command: " + cmd ) ;
2016-04-27 00:00:00 +00:00
return ;
}
2016-04-27 00:00:00 +00:00
s < < " <b>SUCCESS</b>: Command accepted<br><br> \r \n " ;
2016-05-28 00:00:00 +00:00
s < < " <a href= \" /?page=commands \" >Back to commands list</a><br> \r \n " ;
s < < " <p>You will be redirected in 5 seconds</b> " ;
res . add_header ( " Refresh " , " 5; url=/?page=commands " ) ;
2016-08-09 13:54:47 +03:00
}
2016-04-27 00:00:00 +00:00
2016-04-27 00:00:00 +00:00
void HTTPConnection : : SendReply ( HTTPRes & reply , std : : string & content )
2016-04-14 00:00:00 +00:00
{
2017-02-03 14:13:55 -05:00
reply . add_header ( " X-Frame-Options " , " SAMEORIGIN " ) ;
2016-05-24 00:00:00 +00:00
reply . add_header ( " Content-Type " , " text/html " ) ;
reply . body = content ;
2016-04-27 00:00:00 +00:00
2016-08-26 14:00:00 +00:00
m_SendBuffer = reply . to_string ( ) ;
boost : : asio : : async_write ( * m_Socket , boost : : asio : : buffer ( m_SendBuffer ) ,
2016-04-27 00:00:00 +00:00
std : : bind ( & HTTPConnection : : Terminate , shared_from_this ( ) , std : : placeholders : : _1 ) ) ;
2016-04-14 00:00:00 +00:00
}
2015-11-30 16:44:32 +02:00
HTTPServer : : HTTPServer ( const std : : string & address , int port ) :
2016-09-08 09:19:30 -04:00
m_IsRunning ( false ) , m_Thread ( nullptr ) , m_Work ( m_Service ) ,
2015-11-30 16:44:32 +02:00
m_Acceptor ( m_Service , boost : : asio : : ip : : tcp : : endpoint ( boost : : asio : : ip : : address : : from_string ( address ) , port ) )
2013-12-10 08:03:22 -05:00
{
}
HTTPServer : : ~ HTTPServer ( )
{
Stop ( ) ;
}
void HTTPServer : : Start ( )
{
2016-04-27 00:00:00 +00:00
bool needAuth ; i2p : : config : : GetOption ( " http.auth " , needAuth ) ;
std : : string user ; i2p : : config : : GetOption ( " http.user " , user ) ;
std : : string pass ; i2p : : config : : GetOption ( " http.pass " , pass ) ;
/* generate pass if needed */
if ( needAuth & & pass = = " " ) {
2016-07-27 02:00:00 +00:00
uint8_t random [ 16 ] ;
2016-04-27 00:00:00 +00:00
char alnum [ ] = " 0123456789 "
" ABCDEFGHIJKLMNOPQRSTUVWXYZ "
" abcdefghijklmnopqrstuvwxyz " ;
2016-07-27 02:00:00 +00:00
pass . resize ( sizeof ( random ) ) ;
RAND_bytes ( random , sizeof ( random ) ) ;
for ( size_t i = 0 ; i < sizeof ( random ) ; i + + ) {
pass [ i ] = alnum [ random [ i ] % ( sizeof ( alnum ) - 1 ) ] ;
2016-04-27 00:00:00 +00:00
}
i2p : : config : : SetOption ( " http.pass " , pass ) ;
LogPrint ( eLogInfo , " HTTPServer: password set to " , pass ) ;
}
2016-09-08 09:19:30 -04:00
m_IsRunning = true ;
2015-11-22 17:01:37 -05:00
m_Thread = std : : unique_ptr < std : : thread > ( new std : : thread ( std : : bind ( & HTTPServer : : Run , this ) ) ) ;
2013-12-10 08:03:22 -05:00
m_Acceptor . listen ( ) ;
Accept ( ) ;
}
void HTTPServer : : Stop ( )
{
2016-09-08 09:19:30 -04:00
m_IsRunning = false ;
2013-12-10 08:03:22 -05:00
m_Acceptor . close ( ) ;
m_Service . stop ( ) ;
2016-09-08 09:19:30 -04:00
if ( m_Thread )
{
2016-04-27 00:00:00 +00:00
m_Thread - > join ( ) ;
m_Thread = nullptr ;
}
2013-12-10 08:03:22 -05:00
}
void HTTPServer : : Run ( )
{
2016-09-08 09:19:30 -04:00
while ( m_IsRunning )
{
try
{
m_Service . run ( ) ;
}
catch ( std : : exception & ex )
{
LogPrint ( eLogError , " HTTPServer: runtime exception: " , ex . what ( ) ) ;
}
}
2014-07-16 18:41:40 +02:00
}
2013-12-10 08:03:22 -05:00
void HTTPServer : : Accept ( )
{
2015-11-22 17:01:37 -05:00
auto newSocket = std : : make_shared < boost : : asio : : ip : : tcp : : socket > ( m_Service ) ;
m_Acceptor . async_accept ( * newSocket , boost : : bind ( & HTTPServer : : HandleAccept , this ,
boost : : asio : : placeholders : : error , newSocket ) ) ;
2013-12-10 08:03:22 -05:00
}
2015-11-22 17:01:37 -05:00
void HTTPServer : : HandleAccept ( const boost : : system : : error_code & ecode ,
std : : shared_ptr < boost : : asio : : ip : : tcp : : socket > newSocket )
2013-12-10 08:03:22 -05:00
{
2016-04-27 00:00:00 +00:00
if ( ecode )
2016-09-04 16:42:48 -04:00
{
if ( newSocket ) newSocket - > close ( ) ;
LogPrint ( eLogError , " HTTP Server: error handling accept " , ecode . message ( ) ) ;
if ( ecode ! = boost : : asio : : error : : operation_aborted )
Accept ( ) ;
2016-04-27 00:00:00 +00:00
return ;
2016-09-04 16:42:48 -04:00
}
2016-04-27 00:00:00 +00:00
CreateConnection ( newSocket ) ;
Accept ( ) ;
2014-07-16 18:41:40 +02:00
}
2014-03-30 02:16:23 +04:00
2015-11-22 17:01:37 -05:00
void HTTPServer : : CreateConnection ( std : : shared_ptr < boost : : asio : : ip : : tcp : : socket > newSocket )
2014-03-30 02:16:23 +04:00
{
2015-11-22 17:01:37 -05:00
auto conn = std : : make_shared < HTTPConnection > ( newSocket ) ;
2015-04-04 15:44:29 -04:00
conn - > Receive ( ) ;
2014-03-30 02:16:23 +04:00
}
2016-04-27 00:00:00 +00:00
} // http
} // i2p