mirror of
				https://github.com/PurpleI2P/i2pd.git
				synced 2025-10-20 18:50:20 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			1717 lines
		
	
	
	
		
			52 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1717 lines
		
	
	
	
		
			52 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
| * Copyright (c) 2013-2025, The PurpleI2P Project
 | |
| *
 | |
| * This file is part of Purple i2pd project and licensed under BSD3
 | |
| *
 | |
| * See full license text in LICENSE file at top of project tree
 | |
| */
 | |
| 
 | |
| #include <string.h>
 | |
| #include <stdio.h>
 | |
| #ifdef _MSC_VER
 | |
| #include <stdlib.h>
 | |
| #endif
 | |
| #include <charconv>
 | |
| #include "Base.h"
 | |
| #include "Identity.h"
 | |
| #include "Log.h"
 | |
| #include "Destination.h"
 | |
| #include "ClientContext.h"
 | |
| #include "util.h"
 | |
| #include "SAM.h"
 | |
| 
 | |
| namespace i2p
 | |
| {
 | |
| namespace client
 | |
| {
 | |
| 	SAMSocket::SAMSocket (SAMBridge& owner):
 | |
| 		m_Owner (owner), m_Socket(owner.GetService()), m_Timer (m_Owner.GetService ()),
 | |
| 		m_BufferOffset (0), m_SocketType (SAMSocketType::eSAMSocketTypeUnknown),
 | |
| 		m_IsSilent (false), m_IsAccepting (false), m_IsReceiving (false),
 | |
| 		m_Version (MIN_SAM_VERSION)
 | |
| 	{
 | |
| 	}
 | |
| 
 | |
| 	SAMSocket::~SAMSocket ()
 | |
| 	{
 | |
| 		m_Stream = nullptr;
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::Terminate (const char* reason)
 | |
| 	{
 | |
| 		if(m_Stream)
 | |
| 		{
 | |
| 			m_Stream->AsyncClose ();
 | |
| 			m_Stream = nullptr;
 | |
| 		}
 | |
| 		switch (m_SocketType)
 | |
| 		{
 | |
| 			case SAMSocketType::eSAMSocketTypeSession:
 | |
| 				m_Owner.CloseSession (m_ID);
 | |
| 			break;
 | |
| 			case SAMSocketType::eSAMSocketTypeStream:
 | |
| 			break;
 | |
| 			case SAMSocketType::eSAMSocketTypeAcceptor:
 | |
| 			case SAMSocketType::eSAMSocketTypeForward:
 | |
| 			{
 | |
| 				auto session = m_Owner.FindSession(m_ID);
 | |
| 				if (session)
 | |
| 				{
 | |
| 					if (m_IsAccepting && session->GetLocalDestination ())
 | |
| 						session->GetLocalDestination ()->StopAcceptingStreams ();
 | |
| 				}
 | |
| 				break;
 | |
| 			}
 | |
| 			default: ;
 | |
| 		}
 | |
| 		m_SocketType = SAMSocketType::eSAMSocketTypeTerminated;
 | |
| 		if (m_Socket.is_open ())
 | |
| 		{
 | |
| 			boost::system::error_code ec;
 | |
| 			m_Socket.shutdown (boost::asio::ip::tcp::socket::shutdown_both, ec);
 | |
| 			m_Socket.close ();
 | |
| 		}
 | |
| 		m_Owner.RemoveSocket(shared_from_this());
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::ReceiveHandshake ()
 | |
| 	{
 | |
| 		m_Socket.async_read_some (boost::asio::buffer(m_Buffer, SAM_SOCKET_BUFFER_SIZE),
 | |
| 			std::bind(&SAMSocket::HandleHandshakeReceived, shared_from_this (),
 | |
| 			std::placeholders::_1, std::placeholders::_2));
 | |
| 	}
 | |
| 
 | |
| 	static int ExtractVersion (std::string_view ver)
 | |
| 	{
 | |
| 		int version = 0;
 | |
| 		for (auto ch: ver)
 | |
| 		{
 | |
| 			if (ch >= '0' && ch <= '9')
 | |
| 			{
 | |
| 				version *= 10;
 | |
| 				version += (ch - '0');
 | |
| 			}
 | |
| 		}
 | |
| 		return version;
 | |
| 	}	
 | |
| 
 | |
| 	static std::string CreateVersion (int ver)
 | |
| 	{
 | |
| 		auto d = div (ver, 10);
 | |
| 		return std::to_string (d.quot) + "." + std::to_string (d.rem);
 | |
| 	}	
 | |
| 		
 | |
| 	void SAMSocket::HandleHandshakeReceived (const boost::system::error_code& ecode, std::size_t bytes_transferred)
 | |
| 	{
 | |
| 		if (ecode)
 | |
| 		{
 | |
| 			LogPrint (eLogError, "SAM: Handshake read error: ", ecode.message ());
 | |
| 			if (ecode != boost::asio::error::operation_aborted)
 | |
| 				Terminate ("SAM: handshake read error");
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			m_Buffer[bytes_transferred] = 0;
 | |
| 			char * eol = (char *)memchr (m_Buffer, '\n', bytes_transferred);
 | |
| 			if (eol)
 | |
| 				*eol = 0;
 | |
| 			LogPrint (eLogDebug, "SAM: Handshake ", m_Buffer);
 | |
| 			char * separator = strchr (m_Buffer, ' ');
 | |
| 			if (separator)
 | |
| 			{
 | |
| 				separator = strchr (separator + 1, ' ');
 | |
| 				if (separator)
 | |
| 					*separator = 0;
 | |
| 			}
 | |
| 
 | |
| 			if (!strcmp (m_Buffer, SAM_HANDSHAKE))
 | |
| 			{
 | |
| 				int minVer = 0, maxVer = 0;
 | |
| 				// try to find MIN and MAX, 3.0 if not found
 | |
| 				if (separator)
 | |
| 				{
 | |
| 					separator++;
 | |
| 					auto params = ExtractParams (separator);
 | |
| 					auto maxVerStr = params[SAM_PARAM_MAX];
 | |
| 					if (!maxVerStr.empty ())
 | |
| 						maxVer = ExtractVersion (maxVerStr);
 | |
| 					auto minVerStr = params[SAM_PARAM_MIN];
 | |
| 					if (!minVerStr.empty ())
 | |
| 						minVer = ExtractVersion (minVerStr);
 | |
| 				}
 | |
| 				// version negotiation
 | |
| 				if (maxVer && maxVer <= MAX_SAM_VERSION)
 | |
| 					m_Version = maxVer;
 | |
| 				else if (minVer && minVer >= MIN_SAM_VERSION && minVer <= MAX_SAM_VERSION)
 | |
| 					m_Version = minVer;
 | |
| 				else if (!maxVer && !minVer)
 | |
| 					m_Version = MIN_SAM_VERSION;
 | |
| 				else
 | |
| 				{
 | |
| 					LogPrint (eLogError, "SAM: Handshake version mismatch ", minVer, " ", maxVer);
 | |
| 					SendMessageReply (SAM_HANDSHAKE_NOVERSION, true);
 | |
| 					return;
 | |
| 				}	
 | |
| 				// send reply         
 | |
| #ifdef _MSC_VER
 | |
| 				size_t l = sprintf_s (m_Buffer, SAM_SOCKET_BUFFER_SIZE, SAM_HANDSHAKE_REPLY, CreateVersion (m_Version).c_str ());
 | |
| #else
 | |
| 				size_t l = snprintf (m_Buffer, SAM_SOCKET_BUFFER_SIZE, SAM_HANDSHAKE_REPLY, CreateVersion (m_Version).c_str ());
 | |
| #endif
 | |
| 				boost::asio::async_write (m_Socket, boost::asio::buffer (m_Buffer, l), boost::asio::transfer_all (),
 | |
| 					std::bind(&SAMSocket::HandleHandshakeReplySent, shared_from_this (),
 | |
| 					std::placeholders::_1, std::placeholders::_2));
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				LogPrint (eLogError, "SAM: Handshake mismatch");
 | |
| 				Terminate ("SAM: handshake mismatch");
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	bool SAMSocket::IsSession(std::string_view id) const
 | |
| 	{
 | |
| 		return id == m_ID;
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::HandleHandshakeReplySent (const boost::system::error_code& ecode, std::size_t bytes_transferred)
 | |
| 	{
 | |
| 		if (ecode)
 | |
| 		{
 | |
| 			LogPrint (eLogError, "SAM: Handshake reply send error: ", ecode.message ());
 | |
| 			if (ecode != boost::asio::error::operation_aborted)
 | |
| 				Terminate ("SAM: handshake reply send error");
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			m_Socket.async_read_some (boost::asio::buffer(m_Buffer, SAM_SOCKET_BUFFER_SIZE),
 | |
| 				std::bind(&SAMSocket::HandleMessage, shared_from_this (),
 | |
| 				std::placeholders::_1, std::placeholders::_2));
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::SendMessageReply (std::string_view msg, bool close)
 | |
| 	{
 | |
| 		LogPrint (eLogDebug, "SAMSocket::SendMessageReply, close=",close?"true":"false", " reason: ", msg);
 | |
| 
 | |
| 		if (!m_IsSilent || m_SocketType == SAMSocketType::eSAMSocketTypeForward)
 | |
| 			boost::asio::async_write (m_Socket, boost::asio::buffer (msg.data (), msg.size ()), boost::asio::transfer_all (),
 | |
| 				std::bind(&SAMSocket::HandleMessageReplySent, shared_from_this (),
 | |
| 				std::placeholders::_1, std::placeholders::_2, close));
 | |
| 		else
 | |
| 		{
 | |
| 			if (close)
 | |
| 				Terminate ("SAMSocket::SendMessageReply(close=true)");
 | |
| 			else
 | |
| 				Receive ();
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::HandleMessageReplySent (const boost::system::error_code& ecode, std::size_t bytes_transferred, bool close)
 | |
| 	{
 | |
| 		if (ecode)
 | |
| 		{
 | |
| 			LogPrint (eLogError, "SAM: Reply send error: ", ecode.message ());
 | |
| 			if (ecode != boost::asio::error::operation_aborted)
 | |
| 				Terminate ("SAM: reply send error");
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			if (close)
 | |
| 				Terminate ("SAMSocket::HandleMessageReplySent(close=true)");
 | |
| 			else
 | |
| 				Receive ();
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::HandleMessage (const boost::system::error_code& ecode, std::size_t bytes_transferred)
 | |
| 	{
 | |
| 		if (ecode)
 | |
| 		{
 | |
| 			LogPrint (eLogError, "SAM: Read error: ", ecode.message ());
 | |
| 			if (ecode != boost::asio::error::operation_aborted)
 | |
| 				Terminate ("SAM: read error");
 | |
| 		}
 | |
| 		else if (m_SocketType == SAMSocketType::eSAMSocketTypeStream)
 | |
| 			HandleReceived (ecode, bytes_transferred);
 | |
| 		else
 | |
| 		{
 | |
| 			bytes_transferred += m_BufferOffset;
 | |
| 			m_BufferOffset = 0;
 | |
| 			m_Buffer[bytes_transferred] = 0;
 | |
| 			char * eol = (char *)memchr (m_Buffer, '\n', bytes_transferred);
 | |
| 			if (eol)
 | |
| 			{
 | |
| 				if (eol > m_Buffer && eol[-1] == '\r') eol--;
 | |
| 				*eol = 0;
 | |
| 				char * separator = strchr (m_Buffer, ' ');
 | |
| 				if (separator)
 | |
| 				{
 | |
| 					// one word command
 | |
| 					if (std::string_view (m_Buffer, separator - m_Buffer) == SAM_PING)
 | |
| 					{
 | |
| 						ProcessPing ({ separator + 1, (size_t)(eol - separator - 1) });
 | |
| 						return;
 | |
| 					}
 | |
| 
 | |
| 					// two words command
 | |
| 					size_t l = 0;
 | |
| 					separator = strchr (separator + 1, ' ');
 | |
| 					if (separator)
 | |
| 					{	
 | |
| 						*separator = 0;
 | |
| 						l = eol - separator - 1;
 | |
| 					}	
 | |
| 					else
 | |
| 						separator = eol;
 | |
| 
 | |
| 					if (!strcmp (m_Buffer, SAM_SESSION_CREATE))
 | |
| 						ProcessSessionCreate ({ separator + 1, l });
 | |
| 					else if (!strcmp (m_Buffer, SAM_STREAM_CONNECT))
 | |
| 						ProcessStreamConnect (separator + 1, bytes_transferred - (separator - m_Buffer) - 1, bytes_transferred - (eol - m_Buffer) - 1);
 | |
| 					else if (!strcmp (m_Buffer, SAM_STREAM_ACCEPT))
 | |
| 						ProcessStreamAccept ({ separator + 1, l });
 | |
| 					else if (!strcmp (m_Buffer, SAM_STREAM_FORWARD))
 | |
| 						ProcessStreamForward ({ separator + 1, l });
 | |
| 					else if (!strcmp (m_Buffer, SAM_DEST_GENERATE))
 | |
| 						ProcessDestGenerate ({ separator + 1, l });
 | |
| 					else if (!strcmp (m_Buffer, SAM_NAMING_LOOKUP))
 | |
| 						ProcessNamingLookup ({ separator + 1, l });
 | |
| 					else if (!strcmp (m_Buffer, SAM_SESSION_ADD))
 | |
| 						ProcessSessionAdd ({ separator + 1, l });
 | |
| 					else if (!strcmp (m_Buffer, SAM_SESSION_REMOVE))
 | |
| 						ProcessSessionRemove ({ separator + 1, l });
 | |
| 					else if (!strcmp (m_Buffer, SAM_DATAGRAM_SEND) || !strcmp (m_Buffer, SAM_RAW_SEND))
 | |
| 					{
 | |
| 						size_t len = bytes_transferred - (separator - m_Buffer) - 1;
 | |
| 						size_t processed = ProcessDatagramSend (separator + 1, len, eol + 1);
 | |
| 						if (processed < len)
 | |
| 						{
 | |
| 							m_BufferOffset = len - processed;
 | |
| 							if (processed > 0)
 | |
| 								memmove (m_Buffer, separator + 1 + processed, m_BufferOffset);
 | |
| 							else
 | |
| 							{
 | |
| 								// restore string back
 | |
| 								*separator = ' ';
 | |
| 								*eol = '\n';
 | |
| 							}
 | |
| 						}
 | |
| 						// since it's SAM v1 reply is not expected
 | |
| 						Receive ();
 | |
| 					}
 | |
| 					else
 | |
| 					{
 | |
| 						LogPrint (eLogError, "SAM: Unexpected message ", m_Buffer);
 | |
| 						Terminate ("SAM: unexpected message");
 | |
| 					}
 | |
| 				}
 | |
| 				else
 | |
| 				{
 | |
| 					LogPrint (eLogError, "SAM: Malformed message ", m_Buffer);
 | |
| 					Terminate ("malformed message");
 | |
| 				}
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				LogPrint (eLogWarning, "SAM: Incomplete message ", bytes_transferred);
 | |
| 				m_BufferOffset = bytes_transferred;
 | |
| 				// try to receive remaining message
 | |
| 				Receive ();
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	static bool IsAcceptableSessionName(std::string_view str)
 | |
| 	{
 | |
| 		auto itr = str.begin();
 | |
| 		while(itr != str.end())
 | |
| 		{
 | |
| 			char ch = *itr;
 | |
| 			++itr;
 | |
| 			if (ch == '<' || ch == '>' || ch == '"' || ch == '\'' || ch == '/')
 | |
| 				return false;
 | |
| 		}
 | |
| 		return true;
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::ProcessSessionCreate (std::string_view buf)
 | |
| 	{
 | |
| 		LogPrint (eLogDebug, "SAM: Session create: ", buf);
 | |
| 		auto params = ExtractParams (buf);
 | |
| 		std::string_view style = params[SAM_PARAM_STYLE];
 | |
| 		std::string_view id = params[SAM_PARAM_ID];
 | |
| 		std::string_view destination = params[SAM_PARAM_DESTINATION];
 | |
| 
 | |
| 		if(!IsAcceptableSessionName(id))
 | |
| 		{
 | |
| 			// invalid session id
 | |
| 			SendMessageReply (SAM_SESSION_CREATE_INVALID_ID, true);
 | |
| 			return;
 | |
| 		}
 | |
| 		m_ID = id;
 | |
| 		if (m_Owner.FindSession (id))
 | |
| 		{
 | |
| 			// session exists
 | |
| 			SendMessageReply (SAM_SESSION_CREATE_DUPLICATED_ID, true);
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		SAMSessionType type = SAMSessionType::eSAMSessionTypeUnknown;
 | |
| 		i2p::datagram::DatagramVersion datagramVersion = i2p::datagram::eDatagramV1;
 | |
| 		if (style == SAM_VALUE_STREAM) type = SAMSessionType::eSAMSessionTypeStream;
 | |
| #if __cplusplus >= 202002L // C++20
 | |
| 		else if (style.starts_with (SAM_VALUE_DATAGRAM))
 | |
| #else		
 | |
| 		else if (style.substr (0, SAM_VALUE_DATAGRAM.size ()) == SAM_VALUE_DATAGRAM)
 | |
| #endif			
 | |
| 		{	
 | |
| 			// DATAGRAM, DATAGRAM1, DATAGRAM2, DATAGRAM3
 | |
| 			type = SAMSessionType::eSAMSessionTypeDatagram;
 | |
| 			if (style.size () > SAM_VALUE_DATAGRAM.size ())
 | |
| 			{
 | |
| 				switch (style[SAM_VALUE_DATAGRAM.size ()])
 | |
| 				{
 | |
| 					case '1': datagramVersion = i2p::datagram::eDatagramV1; break;
 | |
| 					case '2': datagramVersion = i2p::datagram::eDatagramV2; break;
 | |
| 					case '3': datagramVersion = i2p::datagram::eDatagramV3; break;
 | |
| 					default: type = SAMSessionType::eSAMSessionTypeUnknown;
 | |
| 				}	
 | |
| 			}	
 | |
| 		}	
 | |
| 		else if (style == SAM_VALUE_RAW) type = SAMSessionType::eSAMSessionTypeRaw;
 | |
| 		else if (style == SAM_VALUE_MASTER)
 | |
| 		{	
 | |
| 			if (m_Version < SAM_VERSION_33) // < SAM 3.3
 | |
| 			{
 | |
| 				SendSessionI2PError("MASTER session is not supported");
 | |
| 				return;
 | |
| 			}	
 | |
| 			type = SAMSessionType::eSAMSessionTypeMaster;
 | |
| 		}	
 | |
| 		if (type == SAMSessionType::eSAMSessionTypeUnknown)
 | |
| 		{
 | |
| 			// unknown style
 | |
| 			SendSessionI2PError("Unknown STYLE");
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		std::shared_ptr<boost::asio::ip::udp::endpoint> forward = nullptr;
 | |
| 		if ((type == SAMSessionType::eSAMSessionTypeDatagram || type == SAMSessionType::eSAMSessionTypeRaw) &&
 | |
| 			params.Contains(SAM_PARAM_HOST) && params.Contains(SAM_PARAM_PORT))
 | |
| 		{
 | |
| 			// udp forward selected
 | |
| 			boost::system::error_code e;
 | |
| 			// TODO: support hostnames in udp forward
 | |
| 			auto addr = boost::asio::ip::make_address(params[SAM_PARAM_HOST], e);
 | |
| 			if (e)
 | |
| 			{
 | |
| 				// not an ip address
 | |
| 				SendSessionI2PError("Invalid IP Address in HOST");
 | |
| 				return;
 | |
| 			}
 | |
| 
 | |
| 			uint16_t port = 0;
 | |
| 			std::string_view p = params[SAM_PARAM_PORT];
 | |
| 			auto res = std::from_chars(p.data(), p.data() + p.size(), port);
 | |
| 			if (res.ec != std::errc())
 | |
| 			{
 | |
| 				SendSessionI2PError("Invalid port");
 | |
| 				return;
 | |
| 			}
 | |
| 			forward = std::make_shared<boost::asio::ip::udp::endpoint>(addr, port);
 | |
| 		}
 | |
| 
 | |
| 		//ensure we actually received a destination
 | |
| 		if (destination.empty())
 | |
| 		{
 | |
| 			SendMessageReply (SAM_SESSION_STATUS_INVALID_KEY, true);
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		if (destination != SAM_VALUE_TRANSIENT)
 | |
| 		{
 | |
| 			//ensure it's a base64 string
 | |
| 			i2p::data::PrivateKeys keys;
 | |
| 			if (!keys.FromBase64(destination))
 | |
| 			{
 | |
| 				SendMessageReply(SAM_SESSION_STATUS_INVALID_KEY, true);
 | |
| 				return;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		// create destination
 | |
| 		auto session = m_Owner.CreateSession (id, type, destination == SAM_VALUE_TRANSIENT ? "" : destination, params);
 | |
| 		if (session)
 | |
| 		{
 | |
| 			m_SocketType = SAMSocketType::eSAMSocketTypeSession;
 | |
| 			if (type == SAMSessionType::eSAMSessionTypeDatagram || type == SAMSessionType::eSAMSessionTypeRaw)
 | |
| 			{
 | |
| 				session->UDPEndpoint = forward;
 | |
| 				auto dest = session->GetLocalDestination ()->CreateDatagramDestination (true, datagramVersion);
 | |
| 				uint16_t port = 0;
 | |
| 				if (forward)
 | |
| 				{	
 | |
| 					std::string_view p = params[SAM_PARAM_PORT];
 | |
| 					auto res = std::from_chars(p.data(), p.data() + p.size(), port);
 | |
| 					if (res.ec != std::errc()) port = 0;
 | |
| 				}		
 | |
| 				if (type == SAMSessionType::eSAMSessionTypeDatagram)
 | |
| 					dest->SetReceiver (std::bind (&SAMSocket::HandleI2PDatagramReceive, shared_from_this (),
 | |
| 						std::placeholders::_1, std::placeholders::_2, std::placeholders::_3, 
 | |
| 					    std::placeholders::_4, std::placeholders::_5, std::placeholders::_6),
 | |
| 						port
 | |
| 					);
 | |
| 				else // raw
 | |
| 					dest->SetRawReceiver (std::bind (&SAMSocket::HandleI2PRawDatagramReceive, shared_from_this (),
 | |
| 						std::placeholders::_1, std::placeholders::_2, std::placeholders::_3, std::placeholders::_4),
 | |
| 						port
 | |
| 					);
 | |
| 			}
 | |
| 
 | |
| 			if (session->GetLocalDestination ()->IsReady ())
 | |
| 				SendSessionCreateReplyOk ();
 | |
| 			else
 | |
| 			{
 | |
| 				m_Timer.expires_from_now (boost::posix_time::seconds(SAM_SESSION_READINESS_CHECK_INTERVAL));
 | |
| 				m_Timer.async_wait (std::bind (&SAMSocket::HandleSessionReadinessCheckTimer,
 | |
| 					shared_from_this (), std::placeholders::_1));
 | |
| 			}
 | |
| 		}
 | |
| 		else
 | |
| 			SendMessageReply (SAM_SESSION_CREATE_DUPLICATED_DEST, true);
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::HandleSessionReadinessCheckTimer (const boost::system::error_code& ecode)
 | |
| 	{
 | |
| 		if (ecode != boost::asio::error::operation_aborted)
 | |
| 		{
 | |
| 			if (m_Socket.is_open ())
 | |
| 			{
 | |
| 				auto session = m_Owner.FindSession(m_ID);
 | |
| 				if(session)
 | |
| 				{
 | |
| 					if (session->GetLocalDestination ()->IsReady ())
 | |
| 						SendSessionCreateReplyOk ();
 | |
| 					else
 | |
| 					{
 | |
| 						m_Timer.expires_from_now (boost::posix_time::seconds(SAM_SESSION_READINESS_CHECK_INTERVAL));
 | |
| 						m_Timer.async_wait (std::bind (&SAMSocket::HandleSessionReadinessCheckTimer,
 | |
| 							shared_from_this (), std::placeholders::_1));
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 			else
 | |
| 				Terminate ("SAM: session socket closed");
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::SendSessionCreateReplyOk ()
 | |
| 	{
 | |
| 		auto session = m_Owner.FindSession(m_ID);
 | |
| 		if (session)
 | |
| 		{
 | |
| 			std::string priv = session->GetLocalDestination ()->GetPrivateKeys ().ToBase64 ();
 | |
| #ifdef _MSC_VER
 | |
| 			size_t l2 = sprintf_s (m_Buffer, SAM_SOCKET_BUFFER_SIZE, SAM_SESSION_CREATE_REPLY_OK, priv.c_str ());
 | |
| #else
 | |
| 			size_t l2 = snprintf (m_Buffer, SAM_SOCKET_BUFFER_SIZE, SAM_SESSION_CREATE_REPLY_OK, priv.c_str ());
 | |
| #endif
 | |
| 			SendMessageReply ({m_Buffer, l2}, false);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::ProcessStreamConnect (char * buf, size_t len, size_t rem)
 | |
| 	{
 | |
| 		LogPrint (eLogDebug, "SAM: Stream connect: ", buf);
 | |
| 		if ( m_SocketType != SAMSocketType::eSAMSocketTypeUnknown)
 | |
| 		{
 | |
| 			SendSessionI2PError ("Socket already in use");
 | |
| 			return;
 | |
| 		}
 | |
| 		auto params = ExtractParams (buf);
 | |
| 		std::string_view id = params[SAM_PARAM_ID];
 | |
| 		std::string_view destination = params[SAM_PARAM_DESTINATION];
 | |
| 		std::string_view silent = params[SAM_PARAM_SILENT];
 | |
| 		if (silent == SAM_VALUE_TRUE) m_IsSilent = true;
 | |
| 		m_ID = id;
 | |
| 		auto session = m_Owner.FindSession (id);
 | |
| 		if (session)
 | |
| 		{
 | |
| 			if (rem > 0) // handle follow on data
 | |
| 			{
 | |
| 				memmove (m_Buffer, buf + len + 1, rem); // buf is a pointer to m_Buffer's content
 | |
| 				m_BufferOffset = rem;
 | |
| 			}
 | |
| 			else
 | |
| 				m_BufferOffset = 0;
 | |
| 
 | |
| 			std::shared_ptr<const Address> addr;
 | |
| 			if (destination.find(".i2p") != std::string::npos)
 | |
| 				addr = context.GetAddressBook().GetAddress (destination);
 | |
| 			else
 | |
| 			{
 | |
| 				auto dest = std::make_shared<i2p::data::IdentityEx> ();
 | |
| 				size_t l = dest->FromBase64(destination);
 | |
| 				if (l > 0)
 | |
| 				{
 | |
| 					context.GetAddressBook().InsertFullAddress(dest);
 | |
| 					addr = std::make_shared<Address>(dest->GetIdentHash ());
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			if (addr && addr->IsValid ())
 | |
| 			{
 | |
| 				if (addr->IsIdentHash ())
 | |
| 				{
 | |
| 					if (session->GetLocalDestination ()->GetIdentHash () != addr->identHash)
 | |
| 					{
 | |
| 						auto leaseSet = session->GetLocalDestination ()->FindLeaseSet(addr->identHash);
 | |
| 						if (leaseSet)
 | |
| 							Connect(leaseSet, session);
 | |
| 						else
 | |
| 						{
 | |
| 							session->GetLocalDestination ()->RequestDestination(addr->identHash,
 | |
| 								std::bind(&SAMSocket::HandleConnectLeaseSetRequestComplete,
 | |
| 								shared_from_this(), std::placeholders::_1));
 | |
| 						}
 | |
| 					}
 | |
| 					else
 | |
| 						SendStreamCantReachPeer ("Can't connect to myself");
 | |
| 				}
 | |
| 				else // B33
 | |
| 					session->GetLocalDestination ()->RequestDestinationWithEncryptedLeaseSet (addr->blindedPublicKey,
 | |
| 						std::bind(&SAMSocket::HandleConnectLeaseSetRequestComplete,
 | |
| 						shared_from_this(), std::placeholders::_1));
 | |
| 			}
 | |
| 			else
 | |
| 				SendMessageReply (SAM_STREAM_STATUS_INVALID_KEY, true);
 | |
| 		}
 | |
| 		else
 | |
| 			SendMessageReply (SAM_STREAM_STATUS_INVALID_ID, true);
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::Connect (std::shared_ptr<const i2p::data::LeaseSet> remote, std::shared_ptr<SAMSession> session)
 | |
| 	{
 | |
| 		if (!session) session = m_Owner.FindSession(m_ID);
 | |
| 		if (session)
 | |
| 		{
 | |
| 			if (session->GetLocalDestination ()->SupportsEncryptionType (remote->GetEncryptionType ()))
 | |
| 			{
 | |
| 				m_SocketType = SAMSocketType::eSAMSocketTypeStream;
 | |
| 				m_Stream = session->GetLocalDestination ()->CreateStream (remote);
 | |
| 				if (m_Stream)
 | |
| 				{
 | |
| 					m_Stream->Send ((uint8_t *)m_Buffer, m_BufferOffset); // connect and send
 | |
| 					m_BufferOffset = 0;
 | |
| 					I2PReceive ();
 | |
| 					SendMessageReply (SAM_STREAM_STATUS_OK, false);
 | |
| 				}
 | |
| 				else
 | |
| 					SendMessageReply (SAM_STREAM_STATUS_INVALID_ID, true);
 | |
| 			}
 | |
| 			else
 | |
| 				SendStreamCantReachPeer ("Incompatible crypto");
 | |
| 		}
 | |
| 		else
 | |
| 			SendMessageReply (SAM_STREAM_STATUS_INVALID_ID, true);
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::HandleConnectLeaseSetRequestComplete (std::shared_ptr<i2p::data::LeaseSet> leaseSet)
 | |
| 	{
 | |
| 		if (leaseSet)
 | |
| 			Connect (leaseSet);
 | |
| 		else
 | |
| 		{
 | |
| 			LogPrint (eLogError, "SAM: Destination to connect not found");
 | |
| 			SendStreamCantReachPeer ("LeaseSet not found");
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::ProcessStreamAccept (std::string_view buf)
 | |
| 	{
 | |
| 		LogPrint (eLogDebug, "SAM: Stream accept: ", buf);
 | |
| 		if ( m_SocketType != SAMSocketType::eSAMSocketTypeUnknown)
 | |
| 		{
 | |
| 			SendSessionI2PError ("Socket already in use");
 | |
| 			return;
 | |
| 		}
 | |
| 		auto params = ExtractParams (buf);
 | |
| 		std::string_view id = params[SAM_PARAM_ID];
 | |
| 		std::string_view silent = params[SAM_PARAM_SILENT];
 | |
| 		if (silent == SAM_VALUE_TRUE) m_IsSilent = true;
 | |
| 		m_ID = id;
 | |
| 		auto session = m_Owner.FindSession (id);
 | |
| 		if (session)
 | |
| 		{
 | |
| 			m_SocketType = SAMSocketType::eSAMSocketTypeAcceptor;
 | |
| 			if (!session->GetLocalDestination ()->IsAcceptingStreams ())
 | |
| 			{
 | |
| 				m_IsAccepting = true;
 | |
| 				SendMessageReply (SAM_STREAM_STATUS_OK, false);
 | |
| 				session->GetLocalDestination ()->AcceptOnce (std::bind (&SAMSocket::HandleI2PAccept, shared_from_this (), std::placeholders::_1));
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				auto ts = i2p::util::GetSecondsSinceEpoch ();
 | |
| 				while (!session->acceptQueue.empty () && session->acceptQueue.front ().second + SAM_SESSION_MAX_ACCEPT_INTERVAL > ts)
 | |
| 				{
 | |
| 					auto socket = session->acceptQueue.front ().first;
 | |
| 					session->acceptQueue.pop_front ();
 | |
| 					if (socket)
 | |
| 						boost::asio::post (m_Owner.GetService (), std::bind(&SAMSocket::TerminateClose, socket));
 | |
| 				}
 | |
| 				if (session->acceptQueue.size () < SAM_SESSION_MAX_ACCEPT_QUEUE_SIZE)
 | |
| 				{
 | |
| 					// already accepting, queue up
 | |
| 					SendMessageReply (SAM_STREAM_STATUS_OK, false);
 | |
| 					session->acceptQueue.push_back (std::make_pair(shared_from_this(), ts));
 | |
| 				}
 | |
| 				else
 | |
| 				{
 | |
| 					LogPrint (eLogInfo, "SAM: Session ", m_ID, " accept queue is full ", session->acceptQueue.size ());
 | |
| 					SendStreamI2PError ("Already accepting");
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		else
 | |
| 			SendMessageReply (SAM_STREAM_STATUS_INVALID_ID, true);
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::ProcessStreamForward (std::string_view buf)
 | |
| 	{
 | |
| 		LogPrint(eLogDebug, "SAM: Stream forward: ", buf);
 | |
| 		
 | |
| 		auto params = ExtractParams(buf);
 | |
| 		auto id = params[SAM_PARAM_ID];
 | |
| 		if (id.empty ())
 | |
| 		{
 | |
| 			SendSessionI2PError("Missing ID");
 | |
| 			return;
 | |
| 		}
 | |
| 		
 | |
| 		auto session = m_Owner.FindSession(id);
 | |
| 		if (!session)
 | |
| 		{
 | |
| 			SendMessageReply(SAM_STREAM_STATUS_INVALID_ID, true);
 | |
| 			return;
 | |
| 		}
 | |
| 		if (session->GetLocalDestination()->IsAcceptingStreams())
 | |
| 		{
 | |
| 			SendSessionI2PError("Already accepting");
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		auto portStr = params[SAM_PARAM_PORT];
 | |
| 		if (portStr.empty ())
 | |
| 		{
 | |
| 			SendSessionI2PError("PORT is missing");
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		if (!std::all_of(portStr.begin(), portStr.end(), ::isdigit))
 | |
| 		{
 | |
| 			SendSessionI2PError("Port must be numeric");
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		uint16_t port = 0;
 | |
| 		auto res = std::from_chars(portStr.data(), portStr.data() + portStr.size(), port);
 | |
| 		if (res.ec != std::errc())
 | |
| 		{
 | |
| 			SendSessionI2PError("Invalid port");
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		boost::asio::ip::tcp::endpoint ep;
 | |
| 		auto host = params[SAM_PARAM_HOST];
 | |
| 
 | |
| 		if (!host.empty ())
 | |
| 		{
 | |
| 			boost::system::error_code ec;
 | |
| 			auto addr = boost::asio::ip::make_address(host, ec);
 | |
| 			if (ec)
 | |
| 			{
 | |
| 				SendSessionI2PError("Invalid IP Address in HOST");
 | |
| 				return;
 | |
| 			}
 | |
| 			ep = boost::asio::ip::tcp::endpoint(addr, port);
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			boost::system::error_code ec;
 | |
| 			ep = m_Socket.remote_endpoint(ec);
 | |
| 			if (ec)
 | |
| 			{
 | |
| 				SendSessionI2PError("Socket error: cannot get remote endpoint");
 | |
| 				return;
 | |
| 			}
 | |
| 			ep.port(port);
 | |
| 		}
 | |
| 
 | |
| 		m_SocketType = SAMSocketType::eSAMSocketTypeForward;
 | |
| 		m_ID = id;
 | |
| 		m_IsAccepting = true;
 | |
| 
 | |
| 		if (params[SAM_PARAM_SILENT] == SAM_VALUE_TRUE)
 | |
| 			m_IsSilent = true;
 | |
| 
 | |
| 		session->GetLocalDestination()->AcceptStreams(
 | |
| 			std::bind(&SAMSocket::HandleI2PForward, shared_from_this(), std::placeholders::_1, ep));
 | |
| 
 | |
| 		SendMessageReply(SAM_STREAM_STATUS_OK, false);
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	size_t SAMSocket::ProcessDatagramSend (char * buf, size_t len, const char * data)
 | |
| 	{
 | |
| 		LogPrint (eLogDebug, "SAM: Datagram send: ", buf, " ", len);
 | |
| 		auto params = ExtractParams (buf);
 | |
| 		size_t size = 0;
 | |
| 		std::string_view sizeStr = params[SAM_PARAM_SIZE];
 | |
| 		auto res = std::from_chars(sizeStr.data(), sizeStr.data() + sizeStr.size(), size);
 | |
| 		if (res.ec != std::errc()) size = 0;
 | |
| 		size_t offset = data - buf;
 | |
| 		if (offset + size <= len)
 | |
| 		{
 | |
| 			auto session = m_Owner.FindSession(m_ID);
 | |
| 			if (session)
 | |
| 			{
 | |
| 				auto d = session->GetLocalDestination ()->GetDatagramDestination ();
 | |
| 				if (d)
 | |
| 				{
 | |
| 					i2p::data::IdentityEx dest;
 | |
| 					dest.FromBase64 (params[SAM_PARAM_DESTINATION]);
 | |
| 					if (session->Type == SAMSessionType::eSAMSessionTypeDatagram)
 | |
| 						d->SendDatagramTo ((const uint8_t *)data, size, dest.GetIdentHash ());
 | |
| 					else // raw
 | |
| 						d->SendRawDatagramTo ((const uint8_t *)data, size, dest.GetIdentHash ());
 | |
| 				}
 | |
| 				else
 | |
| 					LogPrint (eLogError, "SAM: Missing datagram destination");
 | |
| 			}
 | |
| 			else
 | |
| 				LogPrint (eLogError, "SAM: Session is not created from DATAGRAM SEND");
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			LogPrint (eLogWarning, "SAM: Sent datagram size ", size, " exceeds buffer ", len - offset);
 | |
| 			return 0; // try to receive more
 | |
| 		}
 | |
| 		return offset + size;
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::ProcessDestGenerate (std::string_view buf)
 | |
| 	{
 | |
| 		LogPrint (eLogDebug, "SAM: Dest generate");
 | |
| 		auto params = ExtractParams (buf);
 | |
| 		// extract signature type
 | |
| 		i2p::data::SigningKeyType signatureType = i2p::data::SIGNING_KEY_TYPE_DSA_SHA1;
 | |
| 		i2p::data::CryptoKeyType cryptoType = i2p::data::CRYPTO_KEY_TYPE_ELGAMAL;
 | |
| 		auto signatureTypeStr = params[SAM_PARAM_SIGNATURE_TYPE];
 | |
| 		if (!signatureTypeStr.empty ())
 | |
| 		{
 | |
| 			if (!m_Owner.ResolveSignatureType (signatureTypeStr, signatureType))
 | |
| 				LogPrint (eLogWarning, "SAM: ", SAM_PARAM_SIGNATURE_TYPE, " is invalid ", signatureTypeStr);
 | |
| 		}
 | |
| 		params.Get (SAM_PARAM_CRYPTO_TYPE, cryptoType);
 | |
| 		auto keys = i2p::data::PrivateKeys::CreateRandomKeys (signatureType, cryptoType, true);
 | |
| #ifdef _MSC_VER
 | |
| 		size_t l = sprintf_s (m_Buffer, SAM_SOCKET_BUFFER_SIZE, SAM_DEST_REPLY,
 | |
| 			keys.GetPublic ()->ToBase64 ().c_str (), keys.ToBase64 ().c_str ());
 | |
| #else
 | |
| 		size_t l = snprintf (m_Buffer, SAM_SOCKET_BUFFER_SIZE, SAM_DEST_REPLY,
 | |
| 			keys.GetPublic ()->ToBase64 ().c_str (), keys.ToBase64 ().c_str ());
 | |
| #endif
 | |
| 		SendMessageReply ({m_Buffer, l}, false);
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::ProcessNamingLookup (std::string_view buf)
 | |
| 	{
 | |
| 		LogPrint (eLogDebug, "SAM: Naming lookup: ", buf);
 | |
| 		auto params = ExtractParams (buf);
 | |
| 		std::string name (params[SAM_PARAM_NAME]);
 | |
| 		std::shared_ptr<const i2p::data::IdentityEx> identity;
 | |
| 		std::shared_ptr<const Address> addr;
 | |
| 		auto session = m_Owner.FindSession(m_ID);
 | |
| 		auto dest = session == nullptr ? context.GetSharedLocalDestination() : session->GetLocalDestination ();
 | |
| 		if (name == "ME")
 | |
| 			SendNamingLookupReply (name, dest->GetIdentity ());
 | |
| 		else if ((identity = context.GetAddressBook ().GetFullAddress (name)) != nullptr)
 | |
| 			SendNamingLookupReply (name, identity);
 | |
| 		else if ((addr = context.GetAddressBook ().GetAddress (name)))
 | |
| 		{
 | |
| 			if (addr->IsIdentHash ())
 | |
| 			{
 | |
| 				auto leaseSet = dest->FindLeaseSet (addr->identHash);
 | |
| 				if (leaseSet)
 | |
| 					SendNamingLookupReply (name, leaseSet->GetIdentity ());
 | |
| 				else
 | |
| 					dest->RequestDestination (addr->identHash,
 | |
| 						std::bind (&SAMSocket::HandleNamingLookupLeaseSetRequestComplete,
 | |
| 						shared_from_this (), std::placeholders::_1, name));
 | |
| 			}
 | |
| 			else
 | |
| 				dest->RequestDestinationWithEncryptedLeaseSet (addr->blindedPublicKey,
 | |
| 					std::bind (&SAMSocket::HandleNamingLookupLeaseSetRequestComplete,
 | |
| 					shared_from_this (), std::placeholders::_1, name));
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			LogPrint (eLogError, "SAM: Naming failed, unknown address ", name);
 | |
| #ifdef _MSC_VER
 | |
| 			size_t len = sprintf_s (m_Buffer, SAM_SOCKET_BUFFER_SIZE, SAM_NAMING_REPLY_INVALID_KEY, name.c_str());
 | |
| #else
 | |
| 			size_t len = snprintf (m_Buffer, SAM_SOCKET_BUFFER_SIZE, SAM_NAMING_REPLY_INVALID_KEY, name.c_str());
 | |
| #endif
 | |
| 			SendMessageReply ({m_Buffer, len}, false);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::ProcessSessionAdd (std::string_view buf)
 | |
| 	{
 | |
| 		if (m_Version < SAM_VERSION_33) // < SAM 3.3
 | |
| 		{
 | |
| 			SendSessionI2PError("SESSION ADD is not supported");
 | |
| 			return;
 | |
| 		}	
 | |
| 		auto session = m_Owner.FindSession(m_ID);
 | |
| 		if (session && session->Type == SAMSessionType::eSAMSessionTypeMaster)
 | |
| 		{
 | |
| 			LogPrint (eLogDebug, "SAM: Subsession add: ", buf);
 | |
| 			auto masterSession = std::static_pointer_cast<SAMMasterSession>(session);
 | |
| 			auto params = ExtractParams (buf);
 | |
| 			std::string_view id = params[SAM_PARAM_ID];
 | |
| 			if (masterSession->subsessions.count (id) > 1)
 | |
| 			{
 | |
| 				// session exists
 | |
| 				SendMessageReply (SAM_SESSION_CREATE_DUPLICATED_ID, false);
 | |
| 				return;
 | |
| 			}
 | |
| 			std::string_view style = params[SAM_PARAM_STYLE];
 | |
| 			SAMSessionType type = SAMSessionType::eSAMSessionTypeUnknown;
 | |
| 			if (style == SAM_VALUE_STREAM) type = SAMSessionType::eSAMSessionTypeStream;
 | |
| 			// TODO: implement other styles
 | |
| 			if (type == SAMSessionType::eSAMSessionTypeUnknown)
 | |
| 			{
 | |
| 				// unknown style
 | |
| 				SendSessionI2PError("Unsupported STYLE");
 | |
| 				return;
 | |
| 			}
 | |
| 			uint16_t fromPort = 0;
 | |
| 			params.Get (SAM_PARAM_FROM_PORT, fromPort);
 | |
| 			
 | |
| 			auto subsession = std::make_shared<SAMSubSession>(masterSession, id, type, fromPort);
 | |
| 			if (m_Owner.AddSession (subsession))
 | |
| 			{
 | |
| 				masterSession->subsessions.insert (std::string (id));
 | |
| 				SendSessionCreateReplyOk ();
 | |
| 			}
 | |
| 			else
 | |
| 				SendMessageReply (SAM_SESSION_CREATE_DUPLICATED_ID, false);
 | |
| 		}
 | |
| 		else
 | |
| 			SendSessionI2PError ("Wrong session type");
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::ProcessSessionRemove (std::string_view buf)
 | |
| 	{
 | |
| 		if (m_Version < SAM_VERSION_33) // < SAM 3.3
 | |
| 		{
 | |
| 			SendSessionI2PError("SESSION REMOVE is not supported");
 | |
| 			return;
 | |
| 		}
 | |
| 		auto session = m_Owner.FindSession(m_ID);
 | |
| 		if (session && session->Type == SAMSessionType::eSAMSessionTypeMaster)
 | |
| 		{
 | |
| 			LogPrint (eLogDebug, "SAM: Subsession remove: ", buf);
 | |
| 			auto masterSession = std::static_pointer_cast<SAMMasterSession>(session);
 | |
| 			auto params = ExtractParams (buf);
 | |
| 			std::string id(params[SAM_PARAM_ID]);
 | |
| 			if (!masterSession->subsessions.erase (id))
 | |
| 			{
 | |
| 				SendMessageReply (SAM_SESSION_STATUS_INVALID_KEY, false);
 | |
| 				return;
 | |
| 			}
 | |
| 			m_Owner.CloseSession (id);
 | |
| 			SendSessionCreateReplyOk ();
 | |
| 		}
 | |
| 		else
 | |
| 			SendSessionI2PError ("Wrong session type");
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::ProcessPing (std::string_view text)
 | |
| 	{
 | |
| 		LogPrint (eLogDebug, "SAM: Ping ", text);
 | |
| 		SendReplyWithMessage (SAM_PONG, std::string (text));
 | |
| 	}	
 | |
| 		
 | |
| 	void SAMSocket::SendReplyWithMessage (const char * reply, const std::string & msg)
 | |
| 	{
 | |
| #ifdef _MSC_VER
 | |
| 		size_t len = sprintf_s (m_Buffer, SAM_SOCKET_BUFFER_SIZE, reply, msg.c_str());
 | |
| #else
 | |
| 		size_t len = snprintf (m_Buffer, SAM_SOCKET_BUFFER_SIZE, reply, msg.c_str());
 | |
| #endif
 | |
| 		SendMessageReply ({m_Buffer, len}, true);
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::SendSessionI2PError(const std::string & msg)
 | |
| 	{
 | |
| 		LogPrint (eLogError, "SAM: Session I2P error: ", msg);
 | |
| 		SendReplyWithMessage (SAM_SESSION_STATUS_I2P_ERROR, msg);
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::SendStreamI2PError(const std::string & msg)
 | |
| 	{
 | |
| 		LogPrint (eLogError, "SAM: Stream I2P error: ", msg);
 | |
| 		SendReplyWithMessage (SAM_STREAM_STATUS_I2P_ERROR, msg);
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::SendStreamCantReachPeer(const std::string & msg)
 | |
| 	{
 | |
| 		SendReplyWithMessage (SAM_STREAM_STATUS_CANT_REACH_PEER, msg);
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::HandleNamingLookupLeaseSetRequestComplete (std::shared_ptr<i2p::data::LeaseSet> leaseSet, std::string name)
 | |
| 	{
 | |
| 		if (leaseSet)
 | |
| 		{
 | |
| 			context.GetAddressBook ().InsertFullAddress (leaseSet->GetIdentity ());
 | |
| 			SendNamingLookupReply (name, leaseSet->GetIdentity ());
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			LogPrint (eLogError, "SAM: Naming lookup failed. LeaseSet for ", name, " not found");
 | |
| #ifdef _MSC_VER
 | |
| 			size_t len = sprintf_s (m_Buffer, SAM_SOCKET_BUFFER_SIZE, SAM_NAMING_REPLY_INVALID_KEY, name.c_str());
 | |
| #else
 | |
| 			size_t len = snprintf (m_Buffer, SAM_SOCKET_BUFFER_SIZE, SAM_NAMING_REPLY_INVALID_KEY, name.c_str());
 | |
| #endif
 | |
| 			SendMessageReply ({m_Buffer, len}, false);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::SendNamingLookupReply (const std::string& name, std::shared_ptr<const i2p::data::IdentityEx> identity)
 | |
| 	{
 | |
| 		auto base64 = identity->ToBase64 ();
 | |
| #ifdef _MSC_VER
 | |
| 		size_t l = sprintf_s (m_Buffer, SAM_SOCKET_BUFFER_SIZE, SAM_NAMING_REPLY, name.c_str (), base64.c_str ());
 | |
| #else
 | |
| 		size_t l = snprintf (m_Buffer, SAM_SOCKET_BUFFER_SIZE, SAM_NAMING_REPLY, name.c_str (), base64.c_str ());
 | |
| #endif
 | |
| 		SendMessageReply ({m_Buffer, l}, false);
 | |
| 	}
 | |
| 
 | |
| 	i2p::util::Mapping SAMSocket::ExtractParams (std::string_view buf)
 | |
| 	{
 | |
| 		i2p::util::Mapping params;
 | |
| 		size_t pos = 0;
 | |
| 		while (pos < buf.length ())
 | |
| 		{	
 | |
| 			std::string_view field;
 | |
| 			auto separator = buf.find (' ', pos);
 | |
| 			if (separator != std::string_view::npos)
 | |
| 			{	
 | |
| 				field = buf.substr (pos, separator - pos);
 | |
| 				pos = separator + 1;
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				field = buf.substr (pos);
 | |
| 				pos = buf.length ();
 | |
| 			}
 | |
| 			auto value = field.find ('=');
 | |
| 			if (value != std::string_view::npos)
 | |
| 				params.Insert (field.substr (0, value), field.substr (value + 1));	
 | |
| 		}
 | |
| 		return params;
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::Receive ()
 | |
| 	{
 | |
| 		if (m_SocketType == SAMSocketType::eSAMSocketTypeStream)
 | |
| 		{
 | |
| 			if (m_IsReceiving) return;
 | |
| 			size_t bufSize = SAM_SOCKET_BUFFER_SIZE;
 | |
| 			size_t unsentSize = m_Stream ? m_Stream->GetSendBufferSize () : 0;
 | |
| 			if (unsentSize)
 | |
| 			{	
 | |
| 				if (unsentSize >= SAM_STREAM_MAX_SEND_BUFFER_SIZE) return; // buffer is full
 | |
| 				if (unsentSize > SAM_STREAM_MAX_SEND_BUFFER_SIZE - SAM_SOCKET_BUFFER_SIZE)
 | |
| 					bufSize = SAM_STREAM_MAX_SEND_BUFFER_SIZE - unsentSize;
 | |
| 			}
 | |
| 			m_IsReceiving = true;
 | |
| 			m_Socket.async_read_some (boost::asio::buffer(m_Buffer, bufSize),
 | |
| 				std::bind(&SAMSocket::HandleReceived, shared_from_this (), std::placeholders::_1, std::placeholders::_2));
 | |
| 		}	
 | |
| 		else
 | |
| 			m_Socket.async_read_some (boost::asio::buffer(m_Buffer + m_BufferOffset, SAM_SOCKET_BUFFER_SIZE - m_BufferOffset),
 | |
| 				std::bind(&SAMSocket::HandleMessage, shared_from_this (), std::placeholders::_1, std::placeholders::_2));
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::HandleReceived (const boost::system::error_code& ecode, std::size_t bytes_transferred)
 | |
| 	{
 | |
| 		m_IsReceiving = false;
 | |
| 		if (ecode)
 | |
| 		{
 | |
| 			LogPrint (eLogError, "SAM: Read error: ", ecode.message ());
 | |
| 			if (ecode != boost::asio::error::operation_aborted)
 | |
| 				Terminate ("read error");
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			if (m_Stream)
 | |
| 			{	
 | |
| 				m_Stream->AsyncSend ((uint8_t *)m_Buffer, bytes_transferred,
 | |
| 					std::bind(&SAMSocket::HandleStreamSend, shared_from_this(), std::placeholders::_1));
 | |
| 				Receive ();
 | |
| 			}	
 | |
| 			else
 | |
| 				Terminate("No Stream Remaining");
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::I2PReceive ()
 | |
| 	{
 | |
| 		if (m_Stream)
 | |
| 		{
 | |
| 			if (m_Stream->GetStatus () == i2p::stream::eStreamStatusNew ||
 | |
| 				m_Stream->GetStatus () == i2p::stream::eStreamStatusOpen) // regular
 | |
| 			{
 | |
| 				m_Stream->AsyncReceive (boost::asio::buffer (m_StreamBuffer, SAM_STREAM_BUFFER_SIZE),
 | |
| 						std::bind (&SAMSocket::HandleI2PReceive, shared_from_this(),
 | |
| 						std::placeholders::_1, std::placeholders::_2),
 | |
| 							SAM_SOCKET_CONNECTION_MAX_IDLE);
 | |
| 			}
 | |
| 			else // closed by peer
 | |
| 			{
 | |
| 				uint8_t * buff = new uint8_t[SAM_STREAM_BUFFER_SIZE];
 | |
| 				// get remaining data
 | |
| 				auto len = m_Stream->ReadSome (buff, SAM_STREAM_BUFFER_SIZE);
 | |
| 				if (len > 0) // still some data
 | |
| 				{
 | |
| 					WriteI2PDataImmediate(buff, len);
 | |
| 				}
 | |
| 				else // no more data
 | |
| 				{
 | |
| 					delete [] buff;
 | |
| 					Terminate ("no more data");
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::WriteI2PDataImmediate(uint8_t * buff, size_t sz)
 | |
| 	{
 | |
| 		boost::asio::async_write (
 | |
| 			m_Socket,
 | |
| 			boost::asio::buffer (buff, sz),
 | |
| 			boost::asio::transfer_all(),
 | |
| 			std::bind (&SAMSocket::HandleWriteI2PDataImmediate, shared_from_this (), std::placeholders::_1, buff)); // postpone termination
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::HandleWriteI2PDataImmediate(const boost::system::error_code & ec, uint8_t * buff)
 | |
| 	{
 | |
| 		delete [] buff;
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::WriteI2PData(size_t sz)
 | |
| 	{
 | |
| 		boost::asio::async_write (
 | |
| 			m_Socket,
 | |
| 			boost::asio::buffer (m_StreamBuffer, sz),
 | |
| 			boost::asio::transfer_all(),
 | |
| 			std::bind(&SAMSocket::HandleWriteI2PData, shared_from_this(), std::placeholders::_1, std::placeholders::_2));
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::HandleI2PReceive (const boost::system::error_code& ecode, std::size_t bytes_transferred)
 | |
| 	{
 | |
| 		if (ecode)
 | |
| 		{
 | |
| 			LogPrint (eLogError, "SAM: Stream read error: ", ecode.message ());
 | |
| 			if (ecode != boost::asio::error::operation_aborted)
 | |
| 			{
 | |
| 				if (bytes_transferred > 0)
 | |
| 				{
 | |
| 					WriteI2PData(bytes_transferred);
 | |
| 				}
 | |
| 				else
 | |
| 				{
 | |
| 					auto s = shared_from_this ();
 | |
| 					boost::asio::post (m_Owner.GetService (), [s] { s->Terminate ("stream read error"); });
 | |
| 				}
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				auto s = shared_from_this ();
 | |
| 				boost::asio::post (m_Owner.GetService (), [s] { s->Terminate ("stream read error (op aborted)"); });
 | |
| 			}
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			if (m_SocketType != SAMSocketType::eSAMSocketTypeTerminated)
 | |
| 			{
 | |
| 				if (bytes_transferred > 0)
 | |
| 				{
 | |
| 					WriteI2PData(bytes_transferred);
 | |
| 				}
 | |
| 				else
 | |
| 					I2PReceive();
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::HandleWriteI2PData (const boost::system::error_code& ecode, size_t bytes_transferred)
 | |
| 	{
 | |
| 		if (ecode)
 | |
| 		{
 | |
| 			LogPrint (eLogError, "SAM: Socket write error: ", ecode.message ());
 | |
| 			if (ecode != boost::asio::error::operation_aborted)
 | |
| 				Terminate ("socket write error at HandleWriteI2PData");
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			I2PReceive ();
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::HandleI2PAccept (std::shared_ptr<i2p::stream::Stream> stream)
 | |
| 	{
 | |
| 		if (stream)
 | |
| 		{
 | |
| 			LogPrint (eLogDebug, "SAM: Incoming I2P connection for session ", m_ID);
 | |
| 			m_SocketType = SAMSocketType::eSAMSocketTypeStream;
 | |
| 			m_IsAccepting = false;
 | |
| 			m_Stream = stream;
 | |
| 			context.GetAddressBook ().InsertFullAddress (stream->GetRemoteIdentity ());
 | |
| 			auto session = m_Owner.FindSession (m_ID);
 | |
| 			if (session && !session->acceptQueue.empty ())
 | |
| 			{
 | |
| 				// pending acceptors
 | |
| 				auto ts = i2p::util::GetSecondsSinceEpoch ();
 | |
| 				while (!session->acceptQueue.empty () && session->acceptQueue.front ().second + SAM_SESSION_MAX_ACCEPT_INTERVAL > ts)
 | |
| 				{
 | |
| 					auto socket = session->acceptQueue.front ().first;
 | |
| 					session->acceptQueue.pop_front ();
 | |
| 					if (socket)
 | |
| 						boost::asio::post (m_Owner.GetService (), std::bind(&SAMSocket::TerminateClose, socket));
 | |
| 				}
 | |
| 				if (!session->acceptQueue.empty ())
 | |
| 				{
 | |
| 					auto socket = session->acceptQueue.front ().first;
 | |
| 					session->acceptQueue.pop_front ();
 | |
| 					if (socket && socket->GetSocketType () == SAMSocketType::eSAMSocketTypeAcceptor)
 | |
| 					{
 | |
| 						socket->m_IsAccepting = true;
 | |
| 						session->GetLocalDestination ()->AcceptOnce (std::bind (&SAMSocket::HandleI2PAccept, socket, std::placeholders::_1));
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 			if (!m_IsSilent)
 | |
| 			{			
 | |
| 				if (m_SocketType != SAMSocketType::eSAMSocketTypeTerminated)
 | |
| 				{
 | |
| 					// get remote peer address
 | |
| 					auto ident = std::make_shared<std::string>(stream->GetRemoteIdentity()->ToBase64 ()); // we need to keep it until sent
 | |
| 					ident->push_back ('\n');
 | |
| 					// send remote peer address back to client like received from stream
 | |
| 					boost::asio::async_write (m_Socket, boost::asio::buffer (ident->data (), ident->size ()), boost::asio::transfer_all(),
 | |
| 						[ident, s = shared_from_this ()](const boost::system::error_code& ecode, size_t bytes_transferred)
 | |
| 						{
 | |
| 							s->HandleWriteI2PData (ecode, bytes_transferred);
 | |
| 						});
 | |
| 				}	
 | |
| 			}
 | |
| 			else
 | |
| 				I2PReceive ();
 | |
| 		}
 | |
| 		else
 | |
| 			LogPrint (eLogWarning, "SAM: I2P acceptor has been reset");
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::HandleI2PForward (std::shared_ptr<i2p::stream::Stream> stream,
 | |
| 		boost::asio::ip::tcp::endpoint ep)
 | |
| 	{
 | |
| 		if (stream)
 | |
| 		{
 | |
| 			LogPrint (eLogDebug, "SAM: Incoming forward I2P connection for session ", m_ID);
 | |
| 			auto newSocket = std::make_shared<SAMSocket>(m_Owner);
 | |
| 			newSocket->SetSocketType (SAMSocketType::eSAMSocketTypeStream);
 | |
| 			auto s = shared_from_this ();
 | |
| 			newSocket->GetSocket ().async_connect (ep,
 | |
| 				[s, newSocket, stream](const boost::system::error_code& ecode)
 | |
| 				{
 | |
| 					if (!ecode)
 | |
| 					{
 | |
| 						s->m_Owner.AddSocket (newSocket);
 | |
| 						newSocket->Receive ();
 | |
| 						newSocket->m_Stream = stream;
 | |
| 						newSocket->m_ID = s->m_ID;
 | |
| 						if (!s->m_IsSilent)
 | |
| 						{
 | |
| 							// get remote peer address
 | |
| 							auto dest = stream->GetRemoteIdentity()->ToBase64 ();
 | |
| 							memcpy (newSocket->m_StreamBuffer, dest.c_str (), dest.length ());
 | |
| 							newSocket->m_StreamBuffer[dest.length ()] = '\n';
 | |
| 							newSocket->HandleI2PReceive (boost::system::error_code (),dest.length () + 1); // we send identity like it has been received from stream
 | |
| 						}
 | |
| 						else
 | |
| 							newSocket->I2PReceive ();
 | |
| 					}
 | |
| 					else
 | |
| 						stream->AsyncClose ();
 | |
| 				});
 | |
| 		}
 | |
| 		else
 | |
| 			LogPrint (eLogWarning, "SAM: I2P forward acceptor has been reset");
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::HandleI2PDatagramReceive (const i2p::data::IdentityEx& from, uint16_t fromPort, uint16_t toPort,
 | |
| 		const uint8_t * buf, size_t len, const i2p::util::Mapping * options)
 | |
| 	{
 | |
| 		LogPrint (eLogDebug, "SAM: Datagram received ", len);
 | |
| 		auto base64 = from.ToBase64 ();
 | |
| 		auto session = m_Owner.FindSession(m_ID);
 | |
| 		if(session)
 | |
| 		{
 | |
| 			auto ep = session->UDPEndpoint;
 | |
| 			if (ep)
 | |
| 			{
 | |
| 				// udp forward enabled
 | |
| 				const char lf = '\n';
 | |
| 				// send to remote endpoint, { destination, linefeed, payload }
 | |
| 				m_Owner.SendTo({ {(const uint8_t *)base64.c_str(), base64.size()}, {(const uint8_t *)&lf, 1}, {buf, len} }, *ep);
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| #ifdef _MSC_VER
 | |
| 				size_t l = sprintf_s ((char *)m_StreamBuffer, SAM_STREAM_BUFFER_SIZE, SAM_DATAGRAM_RECEIVED, base64.c_str (), (long unsigned int)len);
 | |
| #else
 | |
| 				size_t l = snprintf ((char *)m_StreamBuffer, SAM_STREAM_BUFFER_SIZE, SAM_DATAGRAM_RECEIVED, base64.c_str (), (long unsigned int)len);
 | |
| #endif
 | |
| 				if (len < SAM_STREAM_BUFFER_SIZE - l)
 | |
| 				{
 | |
| 					memcpy (m_StreamBuffer + l, buf, len);
 | |
| 					WriteI2PData(len + l);
 | |
| 				}
 | |
| 				else
 | |
| 					LogPrint (eLogWarning, "SAM: Received datagram size ", len," exceeds buffer");
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::HandleI2PRawDatagramReceive (uint16_t fromPort, uint16_t toPort, const uint8_t * buf, size_t len)
 | |
| 	{
 | |
| 		LogPrint (eLogDebug, "SAM: Raw datagram received ", len);
 | |
| 		auto session = m_Owner.FindSession(m_ID);
 | |
| 		if(session)
 | |
| 		{
 | |
| 			auto ep = session->UDPEndpoint;
 | |
| 			if (ep)
 | |
| 				// udp forward enabled
 | |
| 				m_Owner.SendTo({ {buf, len} }, *ep);
 | |
| 			else
 | |
| 			{
 | |
| #ifdef _MSC_VER
 | |
| 				size_t l = sprintf_s ((char *)m_StreamBuffer, SAM_STREAM_BUFFER_SIZE, SAM_RAW_RECEIVED, (long unsigned int)len);
 | |
| #else
 | |
| 				size_t l = snprintf ((char *)m_StreamBuffer, SAM_STREAM_BUFFER_SIZE, SAM_RAW_RECEIVED, (long unsigned int)len);
 | |
| #endif
 | |
| 				if (len < SAM_STREAM_BUFFER_SIZE - l)
 | |
| 				{
 | |
| 					memcpy (m_StreamBuffer + l, buf, len);
 | |
| 					WriteI2PData(len + l);
 | |
| 				}
 | |
| 				else
 | |
| 					LogPrint (eLogWarning, "SAM: Received raw datagram size ", len," exceeds buffer");
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	void SAMSocket::HandleStreamSend(const boost::system::error_code & ec)
 | |
| 	{
 | |
| 		boost::asio::post (m_Owner.GetService (), std::bind( !ec ? &SAMSocket::Receive : &SAMSocket::TerminateClose, shared_from_this()));
 | |
| 	}
 | |
| 
 | |
| 	SAMSession::SAMSession (SAMBridge & parent, std::string_view id, SAMSessionType type):
 | |
| 		m_Bridge(parent), Name(id), Type (type), UDPEndpoint(nullptr)
 | |
| 	{
 | |
| 	}
 | |
| 
 | |
| 	void SAMSession::CloseStreams ()
 | |
| 	{
 | |
| 		for(const auto & itr : m_Bridge.ListSockets(Name))
 | |
| 		{
 | |
| 			itr->Terminate(nullptr);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	SAMSingleSession::SAMSingleSession (SAMBridge & parent, std::string_view name, SAMSessionType type, std::shared_ptr<ClientDestination> dest):
 | |
| 		SAMSession (parent, name, type),
 | |
| 		localDestination (dest)
 | |
| 	{
 | |
| 	}
 | |
| 
 | |
| 	SAMSingleSession::~SAMSingleSession ()
 | |
| 	{
 | |
| 		i2p::client::context.DeleteLocalDestination (localDestination);
 | |
| 	}
 | |
| 
 | |
| 	void SAMSingleSession::StopLocalDestination ()
 | |
| 	{
 | |
| 		localDestination->Release ();
 | |
| 		// stop accepting new streams
 | |
| 		localDestination->StopAcceptingStreams ();
 | |
| 		// terminate existing streams
 | |
| 		auto s = localDestination->GetStreamingDestination (); // TODO: take care about datagrams
 | |
| 		if (s) s->Stop ();
 | |
| 	}
 | |
| 
 | |
| 	void SAMMasterSession::Close ()
 | |
| 	{
 | |
| 		SAMSingleSession::Close ();
 | |
| 		for (const auto& it: subsessions)
 | |
| 			m_Bridge.CloseSession (it);
 | |
| 		subsessions.clear ();
 | |
| 	}
 | |
| 
 | |
| 	SAMSubSession::SAMSubSession (std::shared_ptr<SAMMasterSession> master, std::string_view name, SAMSessionType type, uint16_t port):
 | |
| 		SAMSession (master->m_Bridge, name, type), masterSession (master), inPort (port)
 | |
| 	{	
 | |
| 		if (Type == SAMSessionType::eSAMSessionTypeStream && port)
 | |
| 		{
 | |
| 			// additional streaming destination, use default if port is 0
 | |
| 			auto d = masterSession->GetLocalDestination ()->CreateStreamingDestination (inPort);
 | |
| 			if (d) d->Start ();
 | |
| 		}
 | |
| 		// TODO: implement datagrams
 | |
| 	}
 | |
| 
 | |
| 	std::shared_ptr<ClientDestination> SAMSubSession::GetLocalDestination ()
 | |
| 	{
 | |
| 		return masterSession ? masterSession->GetLocalDestination () : nullptr;
 | |
| 	}
 | |
| 
 | |
| 	void SAMSubSession::StopLocalDestination ()
 | |
| 	{
 | |
| 		auto dest = GetLocalDestination ();
 | |
| 		if (dest && Type == SAMSessionType::eSAMSessionTypeStream)
 | |
| 		{
 | |
| 			auto d = dest->RemoveStreamingDestination (inPort);
 | |
| 			if (d) d->Stop ();
 | |
| 		}
 | |
| 		// TODO: implement datagrams
 | |
| 	}
 | |
| 
 | |
| 	SAMBridge::SAMBridge (const std::string& address, uint16_t portTCP, uint16_t portUDP, bool singleThread):
 | |
| 		RunnableService ("SAM"), m_IsSingleThread (singleThread),
 | |
| 		m_Acceptor (GetIOService (), boost::asio::ip::tcp::endpoint(boost::asio::ip::make_address(address), portTCP)),
 | |
| 		m_DatagramEndpoint (boost::asio::ip::make_address(address), (!portUDP) ? portTCP-1 : portUDP), m_DatagramSocket (GetIOService (), m_DatagramEndpoint),
 | |
| 		m_SignatureTypes
 | |
| 		{
 | |
| 			{"DSA_SHA1", i2p::data::SIGNING_KEY_TYPE_DSA_SHA1},
 | |
| 			{"ECDSA_SHA256_P256", i2p::data::SIGNING_KEY_TYPE_ECDSA_SHA256_P256},
 | |
| 			{"ECDSA_SHA384_P384", i2p::data::SIGNING_KEY_TYPE_ECDSA_SHA384_P384},
 | |
| 			{"ECDSA_SHA512_P521", i2p::data::SIGNING_KEY_TYPE_ECDSA_SHA512_P521},
 | |
| 			{"EdDSA_SHA512_Ed25519", i2p::data::SIGNING_KEY_TYPE_EDDSA_SHA512_ED25519},
 | |
| 			{"GOST_GOSTR3411256_GOSTR3410CRYPTOPROA", i2p::data::SIGNING_KEY_TYPE_GOSTR3410_CRYPTO_PRO_A_GOSTR3411_256},
 | |
| 			{"GOST_GOSTR3411512_GOSTR3410TC26A512", i2p::data::SIGNING_KEY_TYPE_GOSTR3410_TC26_A_512_GOSTR3411_512},
 | |
| 			{"RedDSA_SHA512_Ed25519", i2p::data::SIGNING_KEY_TYPE_REDDSA_SHA512_ED25519},
 | |
| 		}
 | |
| 	{
 | |
| 	}
 | |
| 
 | |
| 	SAMBridge::~SAMBridge ()
 | |
| 	{
 | |
| 		if (IsRunning ())
 | |
| 			Stop ();
 | |
| 	}
 | |
| 
 | |
| 	void SAMBridge::Start ()
 | |
| 	{
 | |
| 		Accept ();
 | |
| 		ReceiveDatagram ();
 | |
| 		StartIOService ();
 | |
| 	}
 | |
| 
 | |
| 	void SAMBridge::Stop ()
 | |
| 	{
 | |
| 		try
 | |
| 		{
 | |
| 			m_Acceptor.cancel ();
 | |
| 		}
 | |
| 		catch (const std::exception& ex)
 | |
| 		{
 | |
| 			LogPrint (eLogError, "SAM: Runtime exception: ", ex.what ());
 | |
| 		}
 | |
| 
 | |
| 		decltype(m_Sessions) sessions;
 | |
| 		{
 | |
| 			std::unique_lock<std::mutex> l(m_SessionsMutex);
 | |
| 			m_Sessions.swap (sessions);
 | |
| 		}	
 | |
| 		for (auto& it: sessions)
 | |
| 			it.second->Close ();
 | |
| 		
 | |
| 		StopIOService ();
 | |
| 	}
 | |
| 
 | |
| 	void SAMBridge::Accept ()
 | |
| 	{
 | |
| 		auto newSocket = std::make_shared<SAMSocket>(*this);
 | |
| 		m_Acceptor.async_accept (newSocket->GetSocket(), std::bind (&SAMBridge::HandleAccept, this,
 | |
| 			std::placeholders::_1, newSocket));
 | |
| 	}
 | |
| 
 | |
| 	void SAMBridge::AddSocket(std::shared_ptr<SAMSocket> socket)
 | |
| 	{
 | |
| 		std::unique_lock<std::mutex> lock(m_OpenSocketsMutex);
 | |
| 		m_OpenSockets.push_back(socket);
 | |
| 	}
 | |
| 
 | |
| 	void SAMBridge::RemoveSocket(const std::shared_ptr<SAMSocket> & socket)
 | |
| 	{
 | |
| 		std::unique_lock<std::mutex> lock(m_OpenSocketsMutex);
 | |
| 		m_OpenSockets.remove_if([socket](const std::shared_ptr<SAMSocket> & item) -> bool { return item == socket; });
 | |
| 	}
 | |
| 
 | |
| 	void SAMBridge::HandleAccept(const boost::system::error_code& ecode, std::shared_ptr<SAMSocket> socket)
 | |
| 	{
 | |
| 		if (!ecode)
 | |
| 		{
 | |
| 			boost::system::error_code ec;
 | |
| 			auto ep = socket->GetSocket ().remote_endpoint (ec);
 | |
| 			if (!ec)
 | |
| 			{
 | |
| 				LogPrint (eLogDebug, "SAM: New connection from ", ep);
 | |
| 				AddSocket (socket);
 | |
| 				socket->ReceiveHandshake ();
 | |
| 			}
 | |
| 			else
 | |
| 				LogPrint (eLogError, "SAM: Incoming connection error: ", ec.message ());
 | |
| 		}
 | |
| 		else
 | |
| 			LogPrint (eLogError, "SAM: Accept error: ", ecode.message ());
 | |
| 
 | |
| 		if (ecode != boost::asio::error::operation_aborted)
 | |
| 			Accept ();
 | |
| 	}
 | |
| 
 | |
| 	std::shared_ptr<SAMSession> SAMBridge::CreateSession (std::string_view id, SAMSessionType type,
 | |
| 		std::string_view destination, const i2p::util::Mapping& params)
 | |
| 	{
 | |
| 		std::shared_ptr<ClientDestination> localDestination = nullptr;
 | |
| 		if (destination != "")
 | |
| 		{
 | |
| 			i2p::data::PrivateKeys keys;
 | |
| 			if (!keys.FromBase64 (destination)) return nullptr;
 | |
| 			localDestination = m_IsSingleThread ?
 | |
| 				i2p::client::context.CreateNewLocalDestination (GetIOService (), keys, true, ¶ms) :
 | |
| 				i2p::client::context.CreateNewLocalDestination (keys, true, ¶ms);
 | |
| 		}
 | |
| 		else // transient
 | |
| 		{
 | |
| 			// extract signature type
 | |
| 			i2p::data::SigningKeyType signatureType = i2p::data::SIGNING_KEY_TYPE_DSA_SHA1;
 | |
| 			i2p::data::CryptoKeyType cryptoType = i2p::data::CRYPTO_KEY_TYPE_ELGAMAL;
 | |
| 			if (!params.IsEmpty ())
 | |
| 			{
 | |
| 				auto signatureTypeStr = params[SAM_PARAM_SIGNATURE_TYPE];
 | |
| 				if (!signatureTypeStr.empty ())
 | |
| 				{
 | |
| 					if (!ResolveSignatureType (signatureTypeStr, signatureType))
 | |
| 						LogPrint (eLogWarning, "SAM: ", SAM_PARAM_SIGNATURE_TYPE, " is invalid ", signatureTypeStr);
 | |
| 				}
 | |
| 				params.Get (SAM_PARAM_CRYPTO_TYPE, cryptoType);
 | |
| 			}
 | |
| 			localDestination = m_IsSingleThread ?
 | |
| 				i2p::client::context.CreateNewLocalDestination (GetIOService (), true, signatureType, cryptoType, ¶ms) :
 | |
| 				i2p::client::context.CreateNewLocalDestination (true, signatureType, cryptoType, ¶ms);
 | |
| 		}
 | |
| 		if (localDestination)
 | |
| 		{
 | |
| 			localDestination->Acquire ();
 | |
| 			auto session = (type == SAMSessionType::eSAMSessionTypeMaster) ? std::make_shared<SAMMasterSession>(*this, id, localDestination) :
 | |
| 				std::make_shared<SAMSingleSession>(*this, id, type, localDestination);
 | |
| 			std::unique_lock<std::mutex> l(m_SessionsMutex);
 | |
| 			auto ret = m_Sessions.emplace (id, session);
 | |
| 			if (!ret.second)
 | |
| 				LogPrint (eLogWarning, "SAM: Session ", id, " already exists");
 | |
| 			return ret.first->second;
 | |
| 		}
 | |
| 		return nullptr;
 | |
| 	}
 | |
| 
 | |
| 	bool SAMBridge::AddSession (std::shared_ptr<SAMSession> session)
 | |
| 	{
 | |
| 		if (!session) return false;
 | |
| 		auto ret = m_Sessions.emplace (session->Name, session);
 | |
| 		return ret.second;
 | |
| 	}
 | |
| 
 | |
| 	void SAMBridge::CloseSession (std::string_view id)
 | |
| 	{
 | |
| 		std::shared_ptr<SAMSession> session;
 | |
| 		{
 | |
| 			std::unique_lock<std::mutex> l(m_SessionsMutex);
 | |
| 			auto it = m_Sessions.find (id);
 | |
| 			if (it != m_Sessions.end ())
 | |
| 			{
 | |
| 				session = it->second;
 | |
| 				m_Sessions.erase (it);
 | |
| 			}
 | |
| 		}
 | |
| 		if (session)
 | |
| 		{
 | |
| 			session->StopLocalDestination ();
 | |
| 			session->Close ();
 | |
| 			if (m_IsSingleThread)
 | |
| 				ScheduleSessionCleanupTimer (session); // let all session's streams close
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	void SAMBridge::ScheduleSessionCleanupTimer (std::shared_ptr<SAMSession> session)
 | |
| 	{
 | |
| 		auto timer = std::make_shared<boost::asio::deadline_timer>(GetService ());
 | |
| 		timer->expires_from_now (boost::posix_time::seconds(5)); // postpone destination clean for 5 seconds
 | |
| 		timer->async_wait (std::bind (&SAMBridge::HandleSessionCleanupTimer, this, std::placeholders::_1, session, timer));
 | |
| 	}	
 | |
| 		
 | |
| 	void SAMBridge::HandleSessionCleanupTimer (const boost::system::error_code& ecode,
 | |
| 		std::shared_ptr<SAMSession> session, std::shared_ptr<boost::asio::deadline_timer> timer)
 | |
| 	{
 | |
| 		if (ecode != boost::asio::error::operation_aborted && session)
 | |
| 		{
 | |
| 			auto dest = session->GetLocalDestination ();
 | |
| 			if (dest)
 | |
| 			{
 | |
| 				auto streamingDest = dest->GetStreamingDestination ();
 | |
| 				if (streamingDest)
 | |
| 				{	
 | |
| 					auto numStreams = streamingDest->GetNumStreams ();
 | |
| 					if (numStreams > 0)
 | |
| 					{
 | |
| 						LogPrint (eLogInfo, "SAM: Session ", session->Name, " still has ", numStreams, " streams");
 | |
| 						ScheduleSessionCleanupTimer (session);
 | |
| 					}	
 | |
| 					else
 | |
| 						LogPrint (eLogDebug, "SAM: Session ", session->Name, " terminated");
 | |
| 				}	
 | |
| 			}	
 | |
| 		}	
 | |
| 		// session's destructor is called here unless rescheduled
 | |
| 	}	
 | |
| 		
 | |
| 	std::shared_ptr<SAMSession> SAMBridge::FindSession (std::string_view id) const
 | |
| 	{
 | |
| 		std::unique_lock<std::mutex> l(m_SessionsMutex);
 | |
| 		auto it = m_Sessions.find (id);
 | |
| 		if (it != m_Sessions.end ())
 | |
| 			return it->second;
 | |
| 		return nullptr;
 | |
| 	}
 | |
| 
 | |
| 	std::list<std::shared_ptr<SAMSocket> > SAMBridge::ListSockets(std::string_view id) const
 | |
| 	{
 | |
| 		std::list<std::shared_ptr<SAMSocket > > list;
 | |
| 		{
 | |
| 			std::unique_lock<std::mutex> l(m_OpenSocketsMutex);
 | |
| 			for (const auto & itr : m_OpenSockets)
 | |
| 				if (itr->IsSession(id))
 | |
| 					list.push_back(itr);
 | |
| 		}
 | |
| 		return list;
 | |
| 	}
 | |
| 
 | |
| 	void SAMBridge::SendTo (const std::vector<boost::asio::const_buffer>& bufs, const boost::asio::ip::udp::endpoint& ep)
 | |
| 	{
 | |
| 		m_DatagramSocket.send_to (bufs, ep);
 | |
| 	}
 | |
| 
 | |
| 	void SAMBridge::ReceiveDatagram ()
 | |
| 	{
 | |
| 		m_DatagramSocket.async_receive_from (
 | |
| 			boost::asio::buffer (m_DatagramReceiveBuffer, i2p::datagram::MAX_DATAGRAM_SIZE),
 | |
| 			m_SenderEndpoint,
 | |
| 			std::bind (&SAMBridge::HandleReceivedDatagram, this, std::placeholders::_1, std::placeholders::_2));
 | |
| 	}
 | |
| 
 | |
| 	void SAMBridge::HandleReceivedDatagram (const boost::system::error_code& ecode, std::size_t bytes_transferred)
 | |
| 	{
 | |
| 		if (!ecode)
 | |
| 		{
 | |
| 			m_DatagramReceiveBuffer[bytes_transferred] = 0;
 | |
| 			char * eol = strchr ((char *)m_DatagramReceiveBuffer, '\n');
 | |
| 			if(eol)
 | |
| 			{
 | |
| 				*eol = 0; eol++;
 | |
| 				size_t payloadLen = bytes_transferred - ((uint8_t *)eol - m_DatagramReceiveBuffer);
 | |
| 				LogPrint (eLogDebug, "SAM: Datagram received ", m_DatagramReceiveBuffer," size=", payloadLen);
 | |
| 				char * sessionID = strchr ((char *)m_DatagramReceiveBuffer, ' ');
 | |
| 				if (sessionID)
 | |
| 				{
 | |
| 					sessionID++;
 | |
| 					char * destination = strchr (sessionID, ' ');
 | |
| 					if (destination)
 | |
| 					{
 | |
| 						*destination = 0; destination++;
 | |
| 						auto session = FindSession (sessionID);
 | |
| 						if (session)
 | |
| 						{
 | |
| 							auto localDest = session->GetLocalDestination ();
 | |
| 							auto datagramDest = localDest ? localDest->GetDatagramDestination () : nullptr;
 | |
| 							if (datagramDest)
 | |
| 							{
 | |
| 								i2p::data::IdentityEx dest;
 | |
| 								dest.FromBase64 (destination);
 | |
| 								if (session->Type == SAMSessionType::eSAMSessionTypeDatagram)
 | |
| 									datagramDest->SendDatagramTo ((uint8_t *)eol, payloadLen, dest.GetIdentHash ());
 | |
| 								else if (session->Type == SAMSessionType::eSAMSessionTypeRaw)
 | |
| 									datagramDest->SendRawDatagramTo ((uint8_t *)eol, payloadLen, dest.GetIdentHash ());
 | |
| 								else
 | |
| 									LogPrint (eLogError, "SAM: Unexpected session type ", (int)session->Type, "for session ", sessionID);
 | |
| 							}
 | |
| 							else
 | |
| 								LogPrint (eLogError, "SAM: Datagram destination is not set for session ", sessionID);
 | |
| 						}
 | |
| 						else
 | |
| 							LogPrint (eLogError, "SAM: Session ", sessionID, " not found");
 | |
| 					}
 | |
| 					else
 | |
| 						LogPrint (eLogError, "SAM: Missing destination key");
 | |
| 				}
 | |
| 				else
 | |
| 					LogPrint (eLogError, "SAM: Missing sessionID");
 | |
| 			}
 | |
| 			else
 | |
| 				LogPrint(eLogError, "SAM: Invalid datagram");
 | |
| 			ReceiveDatagram ();
 | |
| 		}
 | |
| 		else
 | |
| 			LogPrint (eLogError, "SAM: Datagram receive error: ", ecode.message ());
 | |
| 	}
 | |
| 
 | |
| 	bool SAMBridge::ResolveSignatureType (std::string_view name, i2p::data::SigningKeyType& type) const
 | |
| 	{
 | |
| 		auto res = std::from_chars(name.data(), name.data() + name.size(), type);
 | |
| 		if (res.ec != std::errc())
 | |
| 		{
 | |
| 			if (res.ec == std::errc::invalid_argument)
 | |
| 			{
 | |
| 				// name is not numeric, resolving
 | |
| 				auto it = m_SignatureTypes.find (name);
 | |
| 				if (it != m_SignatureTypes.end ())
 | |
| 					type = it->second;
 | |
| 				else
 | |
| 					return false;
 | |
| 			}	
 | |
| 			else
 | |
| 				return false;
 | |
| 		}
 | |
| 		// name has been resolved
 | |
| 		return true;
 | |
| 	}
 | |
| }
 | |
| }
 |